CSDN同步

原题链接

简要题意:

给定一棵树,每个点有黑白两种颜色;对每个节点,求出包含当前节点的连通图,使得白点数与黑点数差最小。输出这些值。

F题也这么简单,咳咳,要是我也熬夜打上那么一场。。。可惜没时间打啊

美国佬怎么想的,不能让比赛设置成美国的上午,那我们就是下午了;非要设置成下午,那我们就是半夜。。。

首先,这题一看就是 \(\texttt{dp}\),树形 \(\texttt{dp}\),换根 \(\texttt{dp}\).

下文中,用 \(\texttt{Subtree(i)}\) 表示 \(i\) 的子树包含的所有节点集合。

用 \(\texttt{father(i)}\) 表示 \(i\) 节点的父亲。

用 \(\texttt{colour_i}\) 表示 \(i\) 节点的颜色值,黑为 \(-1\),白为 \(1\).

首先,假定 \(1\) 为根。

用 \(f_i\) 表示,当前联通图包含在以 \(i\) 为根的子树内的答案。

则必然存在:

\[f_i = colour_i + \sum_{x \in Subtree(i)} \max(f_x,0)
\]

因为, \(colour_i\) 是必须包含的,其次是所有子树中的答案统计;负数不统计。

下面考虑一个换根(树形) \(\texttt{dp}\).用 \(g_i\) 表示 整棵树去掉以 \(i\) 为根的子树后(保留 \(i\) 节点)的答案。

\[g_i = colour_i + \max(0,g_{\texttt{father(i)}} + \sum_{x \in Subtree(\texttt{father(i)})}^{i \not = x} \max(f_x,0))
\]

这也是显然的。

你发现这玩意儿似乎是 \(O(n^2)\) 的???

可以,你机智地发现,后面和 \(f_i\) 的状态转移方程长得不是一点点像!

接着,我们来看后面的部分。

\[\sum_{x \in Subtree(\texttt{father(i)})}^{i \not = x} \max(f_x,0) = f_{\texttt{father(i)}} - \max(0,f_i) - colour_{\texttt{father(i)}}
\]

把这个代入 \(g\) 可知:

\[g_i = colour_i + \max(0,g_{\texttt{father(i)}} + f_{\texttt{father(i)}} - \max(0,f_i) - colour_{\texttt{father(i)}})
\]

然后考虑统计答案。

你可能觉得是这样子的:

\[ans_i = f_i + \max(0,g_i)
\]

可是你机智的发现,连样例都过不了!!!

什么鬼?

\(f_i\) 和 \(g_i\) 都没毛病?

\(ans_i\) 似乎 也没什么问题?

你再次环顾了以下 \(f\) 和 \(g\) 的方程。

你机智的发现,两个函数都计算了 \(colour_i\).

所以还要减掉一个!

\[ans_i = f_i + \max(0,g_i - colour_i)
\]

天哪,你告诉我这个树形 dp 不会写???

要是 \(1\) 年前的我,这里肯定是记忆化搜索。

但是,我们就用树形 \(\texttt{dp}\) 写,怎么地!

具体见代码。

时间复杂度:\(O(n)\).

空间复杂度:\(O(n)\).

实际得分:\(100pts\).

#pragma GCC optimize(2)
#include<bits/stdc++.h>
using namespace std; const int N=2e5+1; inline int read(){char ch=getchar();int f=1;while(ch<'0' || ch>'9') {if(ch=='-') f=-f; ch=getchar();}
int x=0;while(ch>='0' && ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x*f;} int n,a[N],fa[N],f[N];
int g[N],ans[N];
vector<int>G[N];
//正常套路,化树为图,随意求根,形成父亲 inline void solve(int dep,int bs) { //bs 是 dep 的父亲,既方便处理父亲,也为后面的 g 做铺垫
fa[dep]=bs; f[dep]=a[dep]?1:-1;
for(int i=0;i<G[dep].size();i++)
if(G[dep][i]!=bs) {
solve(G[dep][i],dep);
if(f[G[dep][i]]>0) f[dep]+=f[G[dep][i]];
} //加上每个儿子维护的子树值即可,巧妙维护
} inline void dfs(int dep) {
int x=fa[dep],t=g[x]+f[x]-(a[x]?1:-1);
if(f[dep]>0) t-=f[dep];
if(t<0) t=0; ans[dep]=f[dep]+t;
g[dep]=t+(a[dep]?1:-1);
for(int i=0;i<G[dep].size();i++)
if(G[dep][i]-fa[dep]) dfs(G[dep][i]);
} //维护 g int main(){
n=read();
for(int i=1;i<=n;i++) a[i]=read();
for(int i=1;i<n;i++) {
int x=read(),y=read();
G[x].push_back(y);
G[y].push_back(x);
}
solve(1,0); g[1]=a[1]?1:-1; ans[1]=f[1];
for(int i=0;i<G[1].size();i++) dfs(G[1][i]);
for(int i=1;i<=n;i++) printf("%d ",ans[i]);
return 0;
}

CF1324F Maximum White Subtree 题解的更多相关文章

  1. CF1324F Maximum White Subtree——换根dp

    换根dp,一般用来解决在无根树上,需要以每个节点为根跑一边dfs的dp问题 我们做两遍dfs 先钦定任意一个点为根 第一遍,算出\(f_i\)表示\(i\)的子树产生的答案,这里,子树指的是以我们钦定 ...

  2. CF1324 --- Maximum White Subtree

    CF1324 --- Maximum White Subtree 题干 You are given a tree consisting of \(n\) vertices. A tree is a c ...

  3. Codeforces 1324F Maximum White Subtree DFS

    题意 给你无根一颗树,每个节点是黑色或白色.对于每一个节点,问包含该节点的权值最大的子树. 子树的权值等于子树中白点的个数减去黑点的个数. 注意,这里的子树指的是树的联通子图. 解题思路 这场就这题卡 ...

  4. Codeforces Round #627 (Div. 3) F - Maximum White Subtree(深度优先搜索)

    题意: n 个点 n - 1 条边的树,问每个点所在所有子树中白黑点数目的最大差. 思路: 白点先由下至上汇集,后由上至下分并. #include <bits/stdc++.h> usin ...

  5. [Leetcode] 1120. Maximum Average Subtree

    Given the root of a binary tree, find the maximum average value of any subtree of that tree. (A subt ...

  6. 【LeetCode】1120. Maximum Average Subtree 解题报告 (C++)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetcod ...

  7. Lintcode400 Maximum Gap solution 题解

    [题目描述] Given an unsorted array, find the maximum difference between the successive elements in its s ...

  8. Codeforces Round #665 (Div. 2) D. Maximum Distributed Tree 题解(贪心+易错)

    题目链接 题目大意 给你一课树,要你给每一条边分权值,每条边的权值大于0,他们的乘积等于k,而且要使得n-1条边1的数量尽可能少,定义 f(u,v)为u到v的边权和求 \(\max \sum_{i=1 ...

  9. CF1092 --- Tree with Maximum Cost

    CF1324 --- Maximum White Subtree 题干 You are given a tree consisting exactly of \(n\) vertices. Tree ...

随机推荐

  1. 【转载】(String)、toString、String.valueOf的区别

    用于个人参考,查看请前往原地址http://blog.csdn.net/springk/article/details/6414017 问题讨论http://bbs.csdn.net/topics/2 ...

  2. 学习日记:Python爬虫-1

    这几天在b站看小甲鱼的python3教程,照着写了个有道翻译的程序 代码中字典data中的内容,用浏览器审查元素,先随便输一个要翻译的,找到跳出来的post的那个网址,看formdata就行了 返回的 ...

  3. c#或者C#.net中的“ToolTip”是“System.Windows.Forms.ToolTip”和“DevComponents.DotNetBar.ToolTip”之间的不明确的引用

    “ToolTip”是“System.Windows.Forms.ToolTip”和“DevComponents.DotNetBar.ToolTip”之间的不明确的引用 ,在编程时,有时候会编译出现不明 ...

  4. 一条SQL在内存结构与后台进程工作机制

    oracle服务器由数据库以及实例组成,数据库由数据文件,控制文件等物理文件组成,实例是由内存结构+后台进程组成,实例又可以看做连接数据库的方式,在我看来就好比一家公司,实例就是一个决策的办公室,大大 ...

  5. 如何通过学校系统漏洞注册到 @edu.cn 邮箱账号?

    此文章仅针对我自己学校的系统进行分析,并不代表所有学校的系统都是如此. 我们学校比较"抠",可能是为了节省学校的带宽资源然后禁止学生注册教育邮箱账号.不过像一部电影所说的那样&qu ...

  6. Angular 1 深度解析:脏数据检查与 angular 性能优化

    TL;DR 脏检查是一种模型到视图的数据映射机制,由 $apply 或 $digest 触发. 脏检查的范围是整个页面,不受区域或组件划分影响 使用尽量简单的绑定表达式提升脏检查执行速度 尽量减少页面 ...

  7. [LeetCode] 1103. Distribute Candies to People 分糖果

    题目: 思路: 本题一开始的思路就是按照流程一步步分下去,算是暴力方法,在官方题解中有利用等差数列进行计算的 这里只记录一下自己的暴力解题方式 只考虑每次分配的糖果数,分配的糖果数为1,2,3,4,5 ...

  8. 峰哥说技术:02-第一个Spring Boot应用程序

    Spring Boot深度课程系列 峰哥说技术—2020庚子年重磅推出.战胜病毒.我们在行动 02第一个Spring Boot应用程序 1.版本要求 集成开发环境:IntelliJ IDEA 2017 ...

  9. XCTF---easyjni的WriteUp

    一.题目来源     题目来源:XCTF的mobile区的easyjni题目.     题目下载地址:题目链接地址 二.解题过程     1.下载好题目后,安装到夜神模拟器中,发现有一个输入框和一个按 ...

  10. 编码的来源于格式简介ANSI、GBK、GB2312、UTF-8、GB18030和 UNICODE

    编码一直是让新手头疼的问题,特别是 GBK.GB2312.UTF-8 这三个比较常见的网页编码的区别,更是让许多新手晕头转向,怎么解释也解释不清楚.但是编码又是那么重要,特别在网页这一块.如果你打出来 ...