Description

  

​   给一棵\(n\)个节点的树,和一个长度同样为\(n\)的非负整数序列\(x_i\)。

  

​   请尝试对每个节点染黑或白两种颜色,并确定一个非负整数权值。

  

​   问是否存在一种方案,使得每个点\(i\)满足其子树内与其同色的点的权值之和恰好为\(x_i\)。

  

​   \(1 \le n \le 1000\)

  

  ​ \(0 \le x_i \le 5000\)

  

  

  

Solution

  

​   一道好题。

  

​   我们自底向上逐步确定权值。假设当前正在考虑\(u\)这个点。

  

​   我们发现,\(u\)本身选黑还是白并不重要,其子树中的点只有两种:颜色和\(u\)相同的点,以及颜色和\(u\)不相同的点。

  

​   能确定的是,前一类的点的权值值和必须恰好等于\(x_u\),而在这个条件下,后者的点的权值值和可以有多种情况。

  

​   回头看一下,对于一种合法方案,单看每一种颜色形成的树的话,每个点\(u\)都要满足“后继”(单看一个颜色形成的树意义下的后继)的\(x\)之和小于等于\(x_u\),这样一来\(u\)的权值可以设为一个恰好的值,使得和调整为\(x_u\)。

  

​   显然我们可以对每个点\(u\)维护一个值\(f_u\),表示在满足与\(u\)颜色相同的点的权值值和恰好为\(x_u\)时,与\(u\)颜色不同的点的权值之和最小是多少。这是一个贪心的思想,由于一种颜色的权值和只能是\(x_u\),因此我们尽量保证另一种颜色的权值之和最小,以最大化这种颜色在之后考虑父亲节点时满足“小于等于”的可能性。

  

​   设\(g_i\)表示当前与\(u\)同色权值值和为\(i\)时,另一个颜色的最小值是多少。

  

​   初始有\(g_0=0, g_{i}=\infty (i>0)\)

  

​   对于每个后继\(v\),转移

\[g_i+f_v\rightarrow g_{i+x_v}\\g_i+x_v\rightarrow g_{i+f_v}
\]

​   注意是非继承转移,每层转移前每个位置都是\(\infty\)。

  

​   最后\(f_u=\min g_i\;\;(0 \le i \le x_u)\)

  

​   如果$f_1=\infty $就无解,否则有解。

    

​   此题关键是想到贪心的那一步。如果不贪心的话,相当于每个点有多个状态(另一个颜色有多种和)等待转移,根本做不了。

  

​  

  

  

  

Code

  

#include <cstdio>
using namespace std;
const int N=1005,X=5005,INF=1e9;
int n,a[N];
int f[N],g1[X],g2[X];
int h[N],tot;
struct Edge{int v,next;}e[N*2];
inline void addEdge(int u,int v){
e[++tot]=(Edge){v,h[u]}; h[u]=tot;
e[++tot]=(Edge){u,h[v]}; h[v]=tot;
}
inline int min(int x,int y){return x<y?x:y;}
void readData(){
scanf("%d",&n);
int fa;
for(int i=2;i<=n;i++){
scanf("%d",&fa);
addEdge(fa,i);
}
for(int i=1;i<=n;i++) scanf("%d",a+i);
}
void dfs(int u,int fa){
bool have=false;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa){
have=true;
dfs(v,u);
}
if(!have){
f[u]=0;
return;
}
for(int i=0;i<=a[u];i++) g1[i]=INF;
g1[0]=0;
for(int i=h[u],v;i;i=e[i].next)
if((v=e[i].v)!=fa){
for(int j=0;j<=a[u];j++) g2[j]=INF;
for(int j=0;j<=a[u]&&(j+a[v]<=a[u]||j+f[v]<=a[u]);j++)
if(g1[j]!=INF){
if(j+a[v]<=a[u])
g2[j+a[v]]=min(g2[j+a[v]],g1[j]+f[v]);
if(j+f[v]<=a[u])
g2[j+f[v]]=min(g2[j+f[v]],g1[j]+a[v]);
}
for(int j=0;j<=a[u];j++) g1[j]=g2[j];
}
f[u]=INF;
for(int i=0;i<=a[u];i++) f[u]=min(f[u],g1[i]);
}
int main(){
readData();
dfs(1,0);
puts(f[1]==INF?"IMPOSSIBLE":"POSSIBLE");
return 0;
}

【ARC083E】Bichrome Tree的更多相关文章

  1. 【ARC083E】Bichrome Tree 树形dp

    Description 有一颗N个节点的树,其中1号节点是整棵树的根节点,而对于第ii个点(2≤i≤N)(2≤i≤N),其父节点为PiPi 对于这棵树上每一个节点Snuke将会钦定一种颜色(黑或白), ...

  2. 【BZOJ】2631: tree LCT

    [题意]给定n个点的树,每个点初始权值为1,m次操作:1.x到y的点加值,2.断一条边并连一条边,保证仍是树,3.x到y的点乘值,4.x到y的点权值和取模.n,m<=10^5. [算法]Link ...

  3. 【BZOJ2212】[Poi2011]Tree Rotations 线段树合并

    [BZOJ2212][Poi2011]Tree Rotations Description Byteasar the gardener is growing a rare tree called Ro ...

  4. 【题解】Digit Tree

    [题解]Digit Tree CodeForces - 716E 呵呵以为是数据结构题然后是淀粉质还行... 题目就是给你一颗有边权的树,问你有多少路径,把路径上的数字顺次写出来,是\(m\)的倍数. ...

  5. 【题解】[P4178 Tree]

    [题解]P4178 Tree 一道点分治模板好题 不知道是不是我见到的题目太少了,为什么这种题目都是暴力开值域的桶QAQ?? 问点对,考虑点分治吧.直接用值域树状数组开下来,统计的时候直接往树状数组里 ...

  6. 【总结】Link-Cut Tree

    这是一篇关于LCT的总结 加删边的好朋友--Link Cut Tree Link-Cut Tree,LCT的全称 可以说是从树剖引出的问题 树剖可以解决静态的修改或查询树的链上信息:那如果图会不断改变 ...

  7. 【BZOJ】1468: Tree(POJ1741) 点分治

    [题意]给定带边权树,求两点距离<=k的点对数.n<=40000. [算法]点分治 [题解]对于一个区域,选择其重心x作为根,则划分出来的每棵子树都是子区域,可以证明至多划分log n次( ...

  8. 【题解】【BT】【Leetcode】Binary Tree Preorder/Inorder/Postorder (Iterative Solution)

    [Inorder Traversal] Given a binary tree, return the inorder traversal of its nodes' values. For exam ...

  9. 【Leetcode】【Easy】Binary Tree Level Order Traversal II

    Given a binary tree, return the bottom-up level order traversal of its nodes' values. (ie, from left ...

随机推荐

  1. Hacknet 玩后感

    这款游戏的主题是黑客模拟.玩家需要帮助雇主搞定各种乱七八糟的需求. 你需要使用各种工具和各种linux命令行来获取对方电脑的Root权限.然后就是各种增删改查... 解密部分设计的太过简单,导致玩的时 ...

  2. mail邮件详解

    基础命令学习目录首页 1.配置   vim /etc/mail.rc文件尾增加以下内容 set from=1968089885@qq.com smtp="smtp.qq.com"s ...

  3. Final发布文案+美工

    团队名称:探路者 1蔺依铭:http://www.cnblogs.com/linym762/(组长) 2张恩聚:http://www.cnblogs.com/zej87/ 3米赫:http://www ...

  4. 2018软工实践—Beta冲刺(2)

    队名 火箭少男100 组长博客 林燊大哥 作业博客 Beta 冲鸭鸭! 成员冲刺阶段情况 林燊(组长) 过去两天完成了哪些任务 协调组内工作 修改前端界面 展示GitHub当日代码/文档签入记录(组内 ...

  5. C++第一次作业

    Github地址点这里

  6. 手机连接wifi 访问本地服务器网站

    手机连本地wifi后访问 http://192.168.155.1:8001/loc 版权声明:本文为博主原创文章,未经博主允许不得转载.

  7. Backlog和冲刺结果以及产品Demo市场调研

    Backlog和第一阶段冲刺结果以及产品Demo 博客停更了一段时间,但是我们团队没有闲着,现在一次性汇报团队工作进度,Backlog和第一阶段冲刺结果以及产品Demo. 在一段时间的分工合作以及调整 ...

  8. Java第二天——标识符命名规则、Java的知识、快捷键的使用、Scanner获取值的常用方法

    1.标识符命名规则 字母.下划线.数字.美元符号($)由这四个部分组成. 标识符=首字母+其他 首字母:字母.下划线.美元符号($) 其他:字母.下划线.数字.美元符号($) 注意: 1.首字母不能为 ...

  9. 【Coursera】线性回归和逻辑回归

    一.线性回归 1.批量梯度下降法 每次对参数进行一次迭代时,都要扫描一遍输入全集 算法可以收敛到局部最优值 当迭代多次之后,每次迭代参数的改变越小 2.随机梯度下降法 对于一个输入样本,对参数进行一次 ...

  10. 电梯调度系统(界面由C图形库编绘)

    电梯调度系统编程 1.编程题目 电梯调度. 2.结对编程组员 黄冠译,刘畅. 3.编程语言 C语言图形库. 4.题目要求: 5.代码运行及结果调试: ① 运行界面为C++图形库支持,开始运行的初始界面 ...