题目描述的实际是一颗二叉树,对于每个结点,要么满叉,要么无叉。

对于一种无解的简单情况,我们搜一遍树找到最浅的叶子结点1和最深的叶子结点2,如果dep[1]<dep[2]-1,则显然无解。

所以现在所有的叶子结点的深度要么是dep[1]和dep[2].

我们可以把所有结点用node[x]标记状态。

node[x]=0表示x的子树下所有叶子结点深度都是dep[2]。

node[x]=1表示x的子树下一部分叶子结点深度是dep[1],一部分是dep[2]。

node[x]=2表示x的子树下所有叶子结点深度都是dep[1]。

现在有,假设x的儿子结点的node值都是1,则一定无解。证明是显然的。

假设x的左儿子结点node值大于右儿子结点的node值,则ans+1,表示需要操作一次。证明也是显然的。

另外node值可以通过一次树形DP得到,所以总复杂度为O(n).

# include <cstdio>
# include <cstring>
# include <cstdlib>
# include <iostream>
# include <vector>
# include <queue>
# include <stack>
# include <map>
# include <set>
# include <cmath>
# include <algorithm>
using namespace std;
# define lowbit(x) ((x)&(-x))
# define pi acos(-1.0)
# define eps 1e-
# define MOD
# define INF
# define mem(a,b) memset(a,b,sizeof(a))
# define FOR(i,a,n) for(int i=a; i<=n; ++i)
# define FO(i,a,n) for(int i=a; i<n; ++i)
# define bug puts("H");
# define lch p<<,l,mid
# define rch p<<|,mid+,r
# define mp make_pair
# define pb push_back
typedef pair<int,int> PII;
typedef vector<int> VI;
# pragma comment(linker, "/STACK:1024000000,1024000000")
typedef long long LL;
int Scan() {
int res=, flag=;
char ch;
if((ch=getchar())=='-') flag=;
else if(ch>=''&&ch<='') res=ch-'';
while((ch=getchar())>=''&&ch<='') res=res*+(ch-'');
return flag?-res:res;
}
void Out(int a) {
if(a<) {putchar('-'); a=-a;}
if(a>=) Out(a/);
putchar(a%+'');
}
const int N=;
//Code begin... int pos, mi=INF, ma=, node[N*], dep[N*], n, ans, son[N][]; void dfs1(int x)
{
if (x>n) mi=min(mi,dep[x]), ma=max(ma,dep[x]);
else {
dep[son[x][]]=dep[son[x][]]=dep[x]+;
dfs1(son[x][]), dfs1(son[x][]);
}
}
bool dfs2(int x)
{
if (x>n) {
if (dep[x]<ma) node[x]=;
else node[x]=;
}
else {
if (dfs2(son[x][])== || dfs2(son[x][])==) return ;
if(node[son[x][]]==&&node[son[x][]]==) return ;
else if (node[son[x][]]==&&node[son[x][]]==) node[x]=;
else if (node[son[x][]]==&&node[son[x][]]==) node[x]=;
else {
if (node[son[x][]]>node[son[x][]]) ++ans;
node[x]=;
}
}
return ;
}
int main ()
{
int u, v;
scanf("%d",&n);
pos=n;
FOR(i,,n) {
scanf("%d%d",&u,&v);
if (u==-) u=++pos;
if (v==-) v=++pos;
son[i][]=u, son[i][]=v;
}
dfs1();
if (mi<ma-||dfs2()==) puts("-1");
else printf("%d\n",ans);
return ;
}

最新文章

  1. px-rem 一个将px转换为rem的工具
  2. maven+springmvc+dubbo+zookeeper
  3. Flex布局教程及属性速查
  4. sql中union和union all的用法
  5. tomcat session cluster
  6. C++全局和静态变量初始化
  7. PHP flush sleep 输出缓存控制详解
  8. ASP.NET实现大文件下载
  9. Redis的WEB界面管理工具phpRedisAdmin
  10. Chp11: Sorting and Searching
  11. javascript 不间断向左滚动图片
  12. BZOJ 1071 [SCOI2007]组队
  13. 初步掌握Yarn的架构及原理(转)
  14. 六、vue如何缓存页面
  15. 模板 m&#250; bǎn
  16. Java线程调度—休眠
  17. 改变input的值不会触发change事件的解决思路
  18. 163邮箱SMTP设置
  19. Android CollapsingToolbarLayout使用介绍
  20. (转)poj算法做题顺序

热门文章

  1. spring data jap操作
  2. mysql索引 b+树
  3. apache+php开发环境搭建步骤
  4. Centos 7 快速安装 Docker
  5. 微信小程序—day02
  6. 用最简单的MVC模式输出内容
  7. ReadyAPI 教程和示例(二)
  8. JMeter录制Web脚本
  9. Python汉诺塔问题递归算法与程序
  10. leetcode-峰值检测