原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ345.html

前言

我真的是越来越菜了,连树形DP都感觉陌生了。

题解

首先,我们来看看在不断生长叶子会发生什么。

第一种:顺着生长方向走。

第二种:在某一个节点的某些子树依次生长,达到他们之间互相消耗的作用。

对于一个子树 x,假设初始情况下,树心在点 x 上,那么在 x 子树中生长若干次之后,树心离开 x,那么我们考虑求出在树心离开 x 之后,能将 x 往 x 子树方向拉的次数的上界 Max[x] 和下界 Min[x] 。

设 size[x] 表示 x 子树的大小。

显然 Max[x] = size[x]-1 。

而 Min[x] 怎么求?

如果 x 是叶子,那么显然 Min[x] = 0; 如果 x 不是,那么设 x 的儿子中 Max 值最大的节点为 y ,设 $s = \sum_{i\neq y} (Max[i]+1)$ 如果 $s \geq Min[y]+1$ ,则 Min[x] = (s-(Min[y]+1)) mod 2 ,否则 Min[x] = Min[y]+1 - s 。

于是我们已经可以得到根节点的答案了。

那么其他节点呢?

其实大同小异,类似于换根dp地从根出发 dp 一边就好了。

代码

#include <bits/stdc++.h>
#define clr(x) memset(x,0,sizeof (x))
#define For(i,a,b) for (int i=a;i<=b;i++)
#define Fod(i,b,a) for (int i=b;i>=a;i--)
#define pb(x) push_back(x)
#define mp(x,y) make_pair(x,y)
#define fi first
#define se second
#define real __zzd001
#define _SEED_ ('C'+'L'+'Y'+'A'+'K'+'I'+'O'+'I')
#define outval(x) printf(#x" = %d\n",x)
#define outvec(x) printf("vec "#x" = ");for (auto _v : x)printf("%d ",_v);puts("")
#define outtag(x) puts("----------"#x"----------")
#define outarr(a,L,R) printf(#a"[%d...%d] = ",L,R);\
For(_v2,L,R)printf("%d ",a[_v2]);puts("");
using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
typedef vector <int> vi;
LL read(){
LL x=0,f=0;
char ch=getchar();
while (!isdigit(ch))
f|=ch=='-',ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
const int N=100005;
int W,T,n;
vector <int> e[N];
int size[N],Min[N];
int fa[N];
void delfa(int x,int pre){
size[x]=1,fa[x]=pre;
for (auto y : e[x])
if (y!=pre)
delfa(y,x),size[x]+=size[y];
for (int i=0;i<(int)e[x].size();i++)
if (e[x][i]==pre){
swap(e[x][i],e[x].back());
e[x].pop_back();
break;
}
}
void dfs(int x){
for (auto y : e[x])
dfs(y);
if (e[x].size()>0){
int s=size[x]-1,mx=0,mi=-1;
for (auto y : e[x])
if (mx<=size[y])
mx=size[y],mi=Min[y]+1;
s-=mx;
if (s>=mi)
Min[x]=(s-mi)&1;
else
Min[x]=mi-s;
}
else
Min[x]=0;
}
int res[N];
void solve(int x,int d,int s=0){
int fsize=n-size[x]-d;
if (max(Min[x],s)<=min(size[x]-1,fsize))
res[x]=1;
else
res[x]=0;
if (e[x].empty())
return;
int len=e[x].size()+1;
vector <int> sum(len,0),mi(len,-1),mx(len,0);
sum[0]=fsize,mx[0]=fsize,mi[0]=s;
for (int i=1;i<len;i++){
int y=e[x][i-1];
sum[i]=sum[i-1]+size[y];
if (mx[i-1]<=size[y])
mx[i]=size[y],mi[i]=Min[y]+1;
else
mx[i]=mx[i-1],mi[i]=mi[i-1];
}
int sumR=0,miR=-1,mxR=0;
for (int i=len-1;i>=1;i--){
int y=e[x][i-1];
int Sum=sum[i-1]+sumR;
int Mx=mxR,Mi=miR;
if (Mx<=mx[i-1])
Mx=mx[i-1],Mi=mi[i-1];
Sum-=Mx;
solve(y,d+1,Sum>=Mi?((Sum-Mi)&1):Mi-Sum);
sumR+=size[y];
if (mxR<=size[y])
mxR=size[y],miR=Min[y]+1;
}
}
void chk(int x,int v){
res[x]&=v;
for (auto y : e[x])
chk(y,v^1);
}
void Solve(){
n=read();
For(i,0,n)
e[i].clear();
For(i,1,n-1){
int x=read(),y=read();
e[x].pb(y),e[y].pb(x);
}
delfa(1,0);
dfs(1);
solve(1,0);
chk(1,n&1);
if (W==3)
printf("%d\n",res[1]);
else {
for (int i=1;i<=n;i++)
putchar('0'+res[i]);
puts("");
}
}
int main(){
W=read(),T=read();
while (T--)
Solve();
return 0;
}

  

UOJ#345. 【清华集训2017】榕树之心 贪心,动态规划的更多相关文章

  1. [清华集训2017]榕树之心[树dp]

    题意 题目链接 分析 首先解决 \(subtask3\) ,我们的策略就是进入子树,然后用其它子树来抵消,注意在子树内还可以抵消. 可以转化为此模型:有一个数列 \(a\) ,每次我们可以选定两个值 ...

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

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

  3. 【UOJ#340】【清华集训2017】小 Y 和恐怖的奴隶主(矩阵快速幂,动态规划)

    [UOJ#340][清华集训2017]小 Y 和恐怖的奴隶主(矩阵快速幂,动态规划) 题面 UOJ 洛谷 题解 考虑如何暴力\(dp\). 设\(f[i][a][b][c]\)表示当前到了第\(i\) ...

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

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

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

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

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

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

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

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

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

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

  9. loj #2325. 「清华集训 2017」小Y和恐怖的奴隶主

    #2325. 「清华集训 2017」小Y和恐怖的奴隶主 内存限制:256 MiB时间限制:2000 ms标准输入输出 题目类型:传统评测方式:文本比较   题目描述 "A fight? Co ...

随机推荐

  1. SQL2012数据库还原失败System.Data.SqlClient.SqlError: 无法执行 BACKUP LOG,因为当前没有数据库备份

    在一次数据库还原的操作过程中遇到如上错误,去百度了下找到了对应的解决方法 第一个: 在还原完整备份时候,在回复状态状态哪里选择 restore with norecovery ,尾日志部分可选可不选, ...

  2. 五十五、linux 编程——TCP 连接和关闭过程及服务器的并发处理

    55.1 TCP 连接和关闭过程 55.1.1 介绍 建立连接的过程就是三次握手的过程:客户端发送 SYN 报文给服务器,服务器回复 SYN+ACK 报文,客户机再发送 ACK 报文. 关闭连接的过程 ...

  3. mysql的The user specified as a definer (”@’%') does not exist 的解决办法

    两种可能: 1.用户权限不够 赋给用户所有权限试试 mysql> grant all privileges on *.* to root@"%" identified by ...

  4. CPU或以太网模块重启DHCP请求

    最近,带了一个实习生,一天: 他:师傅,PLC如何首次分配IP地址啊? 我:不是教过你了吗? 他:我怎么用BOOTP软件,半天没有反应啊? 我:..... 原来,他做实验的CPU已经被分配IP地址了, ...

  5. 在线制作微信跳转浏览器下载app/打开指定页面源码

    微信自动跳转外部浏览器下载app/打开指定页面源码 源码说明: 适用安卓和苹果系统,支持任何网页链接.并且无论链接是否已经被微信拦截,均可实现微信内自动跳转浏览器打开. 生成的跳转链接具有极佳的防拦截 ...

  6. 论文笔记:Fast(er) RCNN

    在 RCNN 初步试水取得成功后,研究人员又迅速跟进,针对 RCNN 中的几点不足提出改进,接连推出了 fast-rcnn 和 faster-rcnn.关于这两篇论文,网上相关的文章实在是多如牛毛,因 ...

  7. 2018-2019-2 网络对抗技术 20165231 Exp2 后门原理与实践

    实验内容 1.使用netcat获取主机操作Shell,cron启动 2.使用socat获取主机操作Shell, 任务计划启动 3.使用MSF meterpreter(或其他软件)生成可执行文件,利用n ...

  8. hihocoder 1505

    hihocoder 1505 题意:给你n个数,让你从n个数中抽两个数,再抽两个数,使得前两个数和后两个数相等 分析:对 i,j,p,q遍历的话时间复杂度会达到o(n4),所以考虑优化p,q 假设分配 ...

  9. SQL总结 连表查询

    连接查询包括合并.内连接.外连接和交叉连接,如果涉及多表查询,了解这些连接的特点很重要. 只有真正了解它们之间的区别,才能正确使用. 1.Union UNION 操作符用于合并两个或多个 SELECT ...

  10. python使用pip 18以上版本离线安装package

    在内网办公环境,常常需要使用离线安装python的软件包. 一般都会先在互联网的电脑上下载,再拷贝到内网办公机器上进行离线安装. 一般来说,我是这样做的: 1.拷贝和外网电脑上版本一致,且32位或64 ...