传送门

树形dp好题啊。

我们用w[i]" role="presentation" style="position: relative;">w[i]w[i]表示以i为根的子树最少可以把在子树外的榕树之心向子树里拉多少距离。

我们令i最大的子树的根为ms" role="presentation" style="position: relative;">msms。

那么有几种情况:

1. size[ms]" role="presentation" style="position: relative;">size[ms]size[ms]过大,用i其它的子树无法把榕树之心拉回到i去,这个时候有:

w[i]=w[ms]−(size[i]−1−size[ms])" role="presentation" style="position: relative;">w[i]=w[ms]−(size[i]−1−size[ms])w[i]=w[ms]−(size[i]−1−size[ms])。

2. size[ms]" role="presentation" style="position: relative;">size[ms]size[ms]不够大,用i其它子树可以把榕树之心拉回去,但有可能会让榕树之心多走一步。

因此有:

w[i]=(size[i]−1)" role="presentation" style="position: relative;">w[i]=(size[i]−1)w[i]=(size[i]−1) mod" role="presentation" style="position: relative;">modmod 2" role="presentation" style="position: relative;">22

这样我们可以推出w[1]~w[n]。

并且判断整棵树的根节点是否能被凑出。

接下来怎么做呢?

对于一个点i,我们需要判断它是否能被凑出。

我们令depth[i]表示i的深度,其中depth[root]=1" role="presentation" style="position: relative;">depth[root]=1depth[root]=1

首先,如果size[roott]−depth[i]" role="presentation" style="position: relative;">size[roott]−depth[i]size[roott]−depth[i]不是偶数,说明除开root~i这条路径外的其它点一定无法全部抵消,一定会剩下一个,因此最后榕树之心拉不到i节点上。

于是size[root]−depth[i]" role="presentation" style="position: relative;">size[root]−depth[i]size[root]−depth[i]应该为偶数,继续讨论。

这时我们需要把root−>i" role="presentation" style="position: relative;">root−>iroot−>i这条路径看成整个树的根节点,然后把其它的都看成自己的子树像之前求w数组时候那样讨论一下就ok了。

代码:

  1. #include<bits/stdc++.h>
  2. #define mod 998244353
  3. #define N 300005
  4. #define rt 1
  5. using namespace std;
  6. inline int read(){
  7. int ans=0;
  8. char ch=getchar();
  9. while(!isdigit(ch))ch=getchar();
  10. while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
  11. return ans;
  12. }
  13. int W,t,n,cnt=0,siz[N],first[N],mx1[N],mx2[N],dep[N],f[N],w[N],ans[N];
  14. struct edge{int v,next;}e[N<<1];
  15. inline void add(int u,int v){e[++cnt].v=v,e[cnt].next=first[u],first[u]=cnt;}
  16. inline void dfs1(int p,int fa){
  17. siz[p]=1,mx1[p]=mx2[p]=0;
  18. for(int i=first[p];i;i=e[i].next){
  19. int v=e[i].v;
  20. if(v==fa)continue;
  21. dep[v]=dep[p]+1,dfs1(v,p),siz[p]+=siz[v];
  22. if(siz[mx1[p]]<siz[v])mx2[p]=mx1[p],mx1[p]=v;
  23. else if(siz[mx2[p]]<siz[v])mx2[p]=v;
  24. }
  25. if(siz[p]-1>=siz[mx1[p]]+w[mx1[p]])w[p]=(siz[p]-1)%2;
  26. else w[p]=w[mx1[p]]-(siz[p]-1-siz[mx1[p]]);
  27. ++w[p];
  28. }
  29. inline void dfs2(int p,int fa){
  30. ans[p]=0;
  31. if((siz[rt]-dep[p])%2==0){
  32. int v=f[p];
  33. if(siz[v]<siz[mx1[p]])v=mx1[p];
  34. if(siz[rt]-dep[p]-siz[v]-w[v]>=0)ans[p]=1;
  35. }
  36. for(int i=first[p];i;i=e[i].next){
  37. int v=e[i].v;
  38. if(v==fa)continue;
  39. f[v]=f[p];
  40. if(v!=mx1[p]&&siz[mx1[p]]>siz[f[v]])f[v]=mx1[p];
  41. else if(siz[mx2[p]]>siz[f[v]])f[v]=mx2[p];
  42. dfs2(v,p);
  43. }
  44. }
  45. int u,v;
  46. int main(){
  47. W=read(),t=read();
  48. while(t--){
  49. n=read(),cnt=0,memset(first,0,sizeof(first));
  50. for(int i=1;i<n;++i)u=read(),v=read(),add(u,v),add(v,u);
  51. dep[1]=1,dfs1(1,1),f[1]=0,dfs2(1,1);
  52. if(W==3)putchar(ans[1]^48);
  53. else for(int i=1;i<=n;++i)putchar(ans[i]^48);
  54. puts("");
  55. }
  56. return 0;
  57. }

2018.09.01 loj#2330. 「清华集训 2017」榕树之心(树形dp)的更多相关文章

  1. [LOJ#2330]「清华集训 2017」榕树之心

    [LOJ#2330]「清华集训 2017」榕树之心 试题描述 深秋.冷风吹散了最后一丝夏日的暑气,也吹落了榕树脚下灌木丛的叶子.相识数年的Evan和Lyra再次回到了小时候见面的茂盛榕树之下.小溪依旧 ...

  2. Loj #2331. 「清华集训 2017」某位歌姬的故事

    Loj #2331. 「清华集训 2017」某位歌姬的故事 IA 是一名会唱歌的女孩子. IOI2018 就要来了,IA 决定给参赛选手们写一首歌,以表达美好的祝愿.这首歌一共有 \(n\) 个音符, ...

  3. Loj #2324. 「清华集训 2017」小 Y 和二叉树

    Loj #2324. 「清华集训 2017」小 Y 和二叉树 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙上, ...

  4. Loj #2321. 「清华集训 2017」无限之环

    Loj #2321. 「清华集训 2017」无限之环 曾经有一款流行的游戏,叫做 *Infinity Loop***,先来简单的介绍一下这个游戏: 游戏在一个 \(n \times m\) 的网格状棋 ...

  5. Loj 2320.「清华集训 2017」生成树计数

    Loj 2320.「清华集训 2017」生成树计数 题目描述 在一个 \(s\) 个点的图中,存在 \(s-n\) 条边,使图中形成了 \(n\) 个连通块,第 \(i\) 个连通块中有 \(a_i\ ...

  6. [LOJ#2329]「清华集训 2017」我的生命已如风中残烛

    [LOJ#2329]「清华集训 2017」我的生命已如风中残烛 试题描述 九条可怜是一个贪玩的女孩子. 这天她在一堵墙钉了 \(n\) 个钉子,第 \(i\) 个钉子的坐标是 \((x_i,y_i)\ ...

  7. [LOJ#2328]「清华集训 2017」避难所

    [LOJ#2328]「清华集训 2017」避难所 试题描述 "B君啊,你当年的伙伴都不在北京了,为什么你还在北京呢?" "大概是因为出了一些事故吧,否则这道题就不叫避难所 ...

  8. [LOJ#2327]「清华集训 2017」福若格斯

    [LOJ#2327]「清华集训 2017」福若格斯 试题描述 小d是4xx9小游戏高手. 有一天,小d发现了一个很经典的小游戏:跳青蛙. 游戏在一个 \(5\) 个格子的棋盘上进行.在游戏的一开始,最 ...

  9. [LOJ#2326]「清华集训 2017」简单数据结构

    [LOJ#2326]「清华集训 2017」简单数据结构 试题描述 参加完IOI2018之后就是姚班面试.而你,由于讨厌物理.并且想成为乔布斯一样的创业家,被成功踢回贵系. 转眼,时间的指针被指向201 ...

随机推荐

  1. RAD 10 C++Builder的bug

    C++Builder的bug 修改一行代码,F9会报错.要clear工程重新完整编译才可以. 新建空白工程是好的. restart computer ok!!! 2)fdquery like this ...

  2. AWK用法整理

    printf "1:2::3:::4::::5" | awk -F '[:]+' '{print $4}' [:]+ 表示以1个或多个 :(冒号)作为分隔符 ip  addr  | ...

  3. jsp常见的指令总结

    一.三个编译指令 1.page指令: 首先,我们要明确一点就是page指令是一个全局指令,针对当前页面,其次我们再来深挖他的功能,它到底有哪些功能那,在我们程序中起到什么作用??? a.语法结构:&l ...

  4. 真验货客户尾缀sql

    '; --select * from TB_ADDBOMWG_LOG; --SELECT * FROM TB_MAN_ROUTING_QM; SELECT * FROM IN_ITEM WHERE I ...

  5. RN项目中关于父子组件的通信

    子组件向父组件传递数据 子控件中在相应的函数中.通过props.coallback的回调通知父组件. 父组件调用callback属性时行 绑定,并在方法中去解析使用获取到的值 . //子控件: < ...

  6. byte,short,int,long数据之间的倍数关系

    基本数据类型 byte   =  -128和127------------------------------------------------------------2的8次方,1个字节 shor ...

  7. Redis事务的简单理解

    Redis事务的命令如下所示: 先以 MULTI 开始一个事务, 然后将多个命令入队到事务中, 最后由 EXEC 命令触发事务, 一并执行事务中的所有命令 示例如下: //开始一个事务 > MU ...

  8. Control(拆点+最大流)

    Control http://acm.hdu.edu.cn/showproblem.php?pid=4289 Time Limit: 2000/1000 MS (Java/Others)    Mem ...

  9. sqlplus下 查看oracle 执行计划

    Microsoft Windows [版本 6.1.7601] 版权所有 (c) Microsoft Corporation.保留所有权利. C:\Users\zhangzheng2 SQL :: C ...

  10. 动态调用WebService方法

      好像很多人做WebService的时候都是直接添加引用的方式,然后调用服务端的方法.这样就个问题,就是每次我服务端添加了方法或者修改了方法后都要更新Web引用,这样比较麻烦.下面给一个不用添加引用 ...