传送门

树形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了。

代码:

#include<bits/stdc++.h>
#define mod 998244353
#define N 300005
#define rt 1
using namespace std;
inline int read(){
    int ans=0;
    char ch=getchar();
    while(!isdigit(ch))ch=getchar();
    while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
    return ans;
}
int W,t,n,cnt=0,siz[N],first[N],mx1[N],mx2[N],dep[N],f[N],w[N],ans[N];
struct edge{int v,next;}e[N<<1];
inline void add(int u,int v){e[++cnt].v=v,e[cnt].next=first[u],first[u]=cnt;}
inline void dfs1(int p,int fa){
    siz[p]=1,mx1[p]=mx2[p]=0;
    for(int i=first[p];i;i=e[i].next){
        int v=e[i].v;
        if(v==fa)continue;
        dep[v]=dep[p]+1,dfs1(v,p),siz[p]+=siz[v];
        if(siz[mx1[p]]<siz[v])mx2[p]=mx1[p],mx1[p]=v;
        else if(siz[mx2[p]]<siz[v])mx2[p]=v;
    }
    if(siz[p]-1>=siz[mx1[p]]+w[mx1[p]])w[p]=(siz[p]-1)%2;
    else w[p]=w[mx1[p]]-(siz[p]-1-siz[mx1[p]]);
    ++w[p];
}
inline void dfs2(int p,int fa){
    ans[p]=0;
    if((siz[rt]-dep[p])%2==0){
        int v=f[p];
        if(siz[v]<siz[mx1[p]])v=mx1[p];
        if(siz[rt]-dep[p]-siz[v]-w[v]>=0)ans[p]=1;
    }
    for(int i=first[p];i;i=e[i].next){
        int v=e[i].v;
        if(v==fa)continue;
        f[v]=f[p];
        if(v!=mx1[p]&&siz[mx1[p]]>siz[f[v]])f[v]=mx1[p];
        else if(siz[mx2[p]]>siz[f[v]])f[v]=mx2[p];
        dfs2(v,p);
    }
}
int u,v;
int main(){
    W=read(),t=read();
    while(t--){
        n=read(),cnt=0,memset(first,0,sizeof(first));
        for(int i=1;i<n;++i)u=read(),v=read(),add(u,v),add(v,u);
        dep[1]=1,dfs1(1,1),f[1]=0,dfs2(1,1);
        if(W==3)putchar(ans[1]^48);
        else for(int i=1;i<=n;++i)putchar(ans[i]^48);
        puts("");
    }
    return 0;
}

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. 高性能JSON框架之FastJson的简单使用

    1.前言 1.1.FastJson的介绍: JSON协议使用方便,越来越流行,JSON的处理器有很多,这里我介绍一下FastJson,FastJson是阿里的开源框架,被不少企业使用,是一个极其优秀的 ...

  2. UITableView cell 半透明效果,改变cell高度时背景不闪的解决方法

    如果直接指定cell.backgroundColor = = [UIColor colorWithRed:255.0/255.0 green:255.0/255.0 blue:255.0/255.0 ...

  3. Vote Disk 和 OCR概述

    Oracle Clusterware由2部分组成,分别是Voting Disk和 OCR. Voting Disk里面记录着节点成员的信息. 如RAC数据库中有哪些节点成员,节点增加或者删除时也同样会 ...

  4. oc NSLog输出格式大全

    本文的内容是总结了一下iOS开发中NSLog输出格式大全,虽然比较基础,但有总结毕竟会各位正在学习iOS开发的朋友们一些小小的帮助. %@                   对象 %d, %i    ...

  5. 迷你MVVM框架 avalonjs 1.3.8发布

    avalon1.3.8主要是在ms-repeat. ms-each. ms-with等循环绑定上做重大性能优化,其次是对一些绑定了事件的指令添加了roolback,让其CG回收更顺畅. 重构ms-re ...

  6. System.Data.SqlClient.SqlError: 尚未备份数据库的日志尾部

    SQL还原时出现下面的错误,System.Data.SqlClient.SqlError: 尚未备份数据库 "***" 的日志尾部.如果该日志包含您不希望丢失的工作,请使用 BAC ...

  7. 吴裕雄 实战PYTHON编程(10)

    import cv2 cv2.namedWindow("frame")cap = cv2.VideoCapture(0)while(cap.isOpened()): ret, im ...

  8. java多线程实例(2)

    public class ThreadDemo05 { public static void main(String args[]) { // 四个售票点应该控制同一个资源 Demo d = new ...

  9. smartos介绍

    https://wiki.smartos.org A Little History 2005年,Sun Microsystems开源了其著名的Unix操作系统Solaris,最终被发布为一个名为Ope ...

  10. tomcat manager

    在点击tomcat manager的时候提示以下内容: You are not authorized to view this page. By default the Host Manager is ...