[清华集训2017]榕树之心[树dp]
时间:2019-03-14
本文章向大家介绍[清华集训2017]榕树之心[树dp],主要包括[清华集训2017]榕树之心[树dp]使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
题意
分析
首先解决 \(subtask3\) ,我们的策略就是进入子树,然后用其它子树来抵消,注意在子树内还可以抵消。
可以转化为此模型:有一个数列 \(a\) ,每次我们可以选定两个值 \(>0\) 的数并让他们同时减 1,让最后 \(S=\sum a\) 最小。
如果最大的数 \(a_{mx}\ge S\) ,显然答案为 \(2*a_{mx}-S\) 。
否则我们每次把最大和次大的两个数进行操作,容易证明最后的答案为 \(S\ \rm mod\ 2\) 。
现在数列的每一项就是子树的大小。
也就是说,如果我们能够尽量多地让重儿子内部相互抵消(假设最多抵消剩 \(x\) 个节点,那么可以保留的节点就是 (\(x+2k+1\))),以至于重儿子剩余的大小 \(\le \frac{S}{2}\) 且保持最大或最大-1(如果只有两棵子树,且初始次大子树比最大子树小1不会影响讨论),就可以让结果得至多为 1 了。
发现这变成了一个子问题,所以记状态 \(f_u\) 表示从 \(u\) 出发再回到 \(u\) 之后最少剩下多少节点。转移就比较显然了。
对于不止查询一号点的情况,给每个点记个最大和次大的儿子, dfs 到 \(u\) 时把路径上的点看成一个点即可。
时间复杂度 \(O(n)\) 。
代码
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
#define go(u) for(int i = head[u], v = e[i].to; i; i=e[i].lst, v=e[i].to)
#define rep(i, a, b) for(int i = a; i <= b; ++i)
#define pb push_back
#define re(x) memset(x, 0, sizeof x)
inline int gi() {
int x = 0,f = 1;
char ch = getchar();
while(!isdigit(ch)) { if(ch == '-') f = -1; ch = getchar();}
while(isdigit(ch)) { x = (x << 3) + (x << 1) + ch - 48; ch = getchar();}
return x * f;
}
template <typename T> inline bool Max(T &a, T b){return a < b ? a = b, 1 : 0;}
template <typename T> inline bool Min(T &a, T b){return a > b ? a = b, 1 : 0;}
const int N = 1e5 + 7, inf = 0x3f3f3f3f;
int n, W, T;
int ans[N], dep[N];
vector<int>G[N];
struct data {
int son, f;
data(){}data(int son, int f):son(son), f(f){}
bool operator <(const data &rhs) const {
if(son != rhs.son) return son < rhs.son;
return f > rhs.f;
}
bool operator ==(const data &rhs) const {
return son == rhs.son && f == rhs.f;
}
}f[N][2], g[N];
void Add(int a, int b) {
G[a].pb(b);
G[b].pb(a);
}
void dfs1(int u, int fa) {
g[u].son = 1;
for(auto v:G[u]) if(v ^ fa) {
dep[v] = dep[u] + 1;
dfs1(v, u);
if(f[u][0] < g[v]) {
f[u][1] = f[u][0];
f[u][0] = g[v];
}else if(f[u][1] < g[v]) {
f[u][1] = g[v];
}
g[u].son += g[v].son;
}
if(f[u][0].son) {
if(g[u].son - f[u][0].son - 1 >= f[u][0].f + 1)
g[u].f = (g[u].son - 1) % 2;
else
g[u].f = f[u][0].f + 1 - (g[u].son - f[u][0].son - 1);
}else g[u].f = 0;
}
void dfs2(int u, int fa, data mx) {
data now = max(mx, f[u][0]);
int son = n - dep[u];
if(now.son) {
if(son - now.son - 1 >= now.f + 1)
ans[u] = (son - 1) % 2;
else
ans[u] = now.f + 1 - (son - now.son - 1);
}
for(auto v:G[u]) if(v ^ fa) {
if(f[u][0] == g[v]) dfs2(v, u, max(mx, f[u][1]));
else dfs2(v, u, max(mx, f[u][0]));
}
}
int main() {
W = gi(), T = gi();
while(T--) {
n = gi();
rep(i, 1, n) G[i].clear(), f[i][0] = f[i][1] = g[i] = data(0, inf);
rep(i, 1, n - 1) {
int a = gi(), b = gi();
Add(a, b);
}
dfs1(1, 0);
dfs2(1, 0, data(0, inf));
if(W == 3)
printf("%d\n", ans[1] == 0);
else
rep(i, 1, n) printf("%d", ans[i] == 0);
puts("");
}
return 0;
}
- Java 机器学习库Smile实战(二)AdaBoost
- 如何用excel urldecode解码把url编码转为汉字?
- Scikit-learn实战之 SVM回归分析、密度估计、异常点检测
- TensorFlow模拟简单线性模型小栗子
- web跨域解决方案
- Bootstrap幻灯轮播如何支持触屏左右滑动手势?
- JavaScript 调试小技巧
- 图片上传预览js
- Isolation Forest算法实现详解
- css继承样式怎么控制?用选择器
- wordpress站内搜索结果页URL伪静态如何操作
- 如何实现大图居中超过的部分两边自动隐藏
- Ubuntu16.04安装后开发环境配置和常用软件安装
- wordpress如何屏蔽wp-json(禁用REST API)
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Python OpenCV查找图中的四边形/矩形
- Spark Opeartor的指标体系
- R-基本绘图参数(Ⅰ)
- NVIDIA TensorRT Inference Server on Kubernetes
- Kubernetes 环境的 Tensorflow Serving on S3
- 数据处理|数据框重铸
- Go mod 常见问题(持续更新)
- R In Action|创建数据集
- R语言进阶之生存分析
- R In Action |基本数据管理
- R|apply,tapply
- python中的基本运算
- 助力联邦——Pulsar在Angel PowerFL联邦学习平台中的应用
- 让数据跃然“图”上!腾讯位置服务数据可视化API正式发布
- 腾讯云 Serverless 衔接 Kafka 上下游数据流转实战