CF1324F Maximum White Subtree 题解
简要题意:
给定一棵树,每个点有黑白两种颜色;对每个节点,求出包含当前节点的连通图,使得白点数与黑点数差最小。输出这些值。
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\) 为根的子树内的答案。
则必然存在:
\]
因为, \(colour_i\) 是必须包含的,其次是所有子树中的答案统计;负数不统计。
下面考虑一个换根(树形) \(\texttt{dp}\).用 \(g_i\) 表示 整棵树去掉以 \(i\) 为根的子树后(保留 \(i\) 节点)的答案。
\]
这也是显然的。
你发现这玩意儿似乎是 \(O(n^2)\) 的???
可以,你机智地发现,后面和 \(f_i\) 的状态转移方程长得不是一点点像!
接着,我们来看后面的部分。
\]
把这个代入 \(g\) 可知:
\]
然后考虑统计答案。
你可能觉得是这样子的:
\]
可是你机智的发现,连样例都过不了!!!
什么鬼?
\(f_i\) 和 \(g_i\) 都没毛病?
\(ans_i\) 似乎 也没什么问题?
你再次环顾了以下 \(f\) 和 \(g\) 的方程。
你机智的发现,两个函数都计算了 \(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 题解的更多相关文章
- CF1324F Maximum White Subtree——换根dp
换根dp,一般用来解决在无根树上,需要以每个节点为根跑一边dfs的dp问题 我们做两遍dfs 先钦定任意一个点为根 第一遍,算出\(f_i\)表示\(i\)的子树产生的答案,这里,子树指的是以我们钦定 ...
- CF1324 --- Maximum White Subtree
CF1324 --- Maximum White Subtree 题干 You are given a tree consisting of \(n\) vertices. A tree is a c ...
- Codeforces 1324F Maximum White Subtree DFS
题意 给你无根一颗树,每个节点是黑色或白色.对于每一个节点,问包含该节点的权值最大的子树. 子树的权值等于子树中白点的个数减去黑点的个数. 注意,这里的子树指的是树的联通子图. 解题思路 这场就这题卡 ...
- Codeforces Round #627 (Div. 3) F - Maximum White Subtree(深度优先搜索)
题意: n 个点 n - 1 条边的树,问每个点所在所有子树中白黑点数目的最大差. 思路: 白点先由下至上汇集,后由上至下分并. #include <bits/stdc++.h> usin ...
- [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 ...
- 【LeetCode】1120. Maximum Average Subtree 解题报告 (C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetcod ...
- Lintcode400 Maximum Gap solution 题解
[题目描述] Given an unsorted array, find the maximum difference between the successive elements in its s ...
- Codeforces Round #665 (Div. 2) D. Maximum Distributed Tree 题解(贪心+易错)
题目链接 题目大意 给你一课树,要你给每一条边分权值,每条边的权值大于0,他们的乘积等于k,而且要使得n-1条边1的数量尽可能少,定义 f(u,v)为u到v的边权和求 \(\max \sum_{i=1 ...
- CF1092 --- Tree with Maximum Cost
CF1324 --- Maximum White Subtree 题干 You are given a tree consisting exactly of \(n\) vertices. Tree ...
随机推荐
- 修改从Maven中心仓库下载到本地的jar包的默认存储位置及远程仓库
从Maven中心仓库下载到本地的jar包的默认存放在”${user.home}/.m2/repository”中,${user.home}表示当前登录系统的用户目录(如"C:\Users\g ...
- 生死状:苹果VS他的供应商
据知情人士透露,苹果已经组建了代号为Titan的汽车团队,并招募了数百名员工,准备进入汽车领域,iCar大有呼之欲出之势.事实上,苹果CEO蒂姆-库克早在去年就参观了宝马位于莱比锡的核心工厂,学习如何 ...
- docker-compose的安装和设定
docker的1.12版本中,swarm已经合体,docker-engine/swarm/docker-compose的三件套装已经变成两件.后续会不会将docker-compose进一步合体呢,想做 ...
- 创建Cocoapods私有库
本文以自己在公司做的一个手势密码私有库GesturePasswordKit为例说明. 1.在gitlab(或者github,我这里使用的例子是在gitlab上)上创建git仓库 (确保授权正确,避免后 ...
- Dubbo源码学习(二)
@Adaptive注解 在上一篇ExtensionLoader的博客中记录了,有两种扩展点,一种是普通的扩展实现,另一种就是自适应的扩展点,即@Adaptive注解的实现类. @Documented ...
- 初识Spring JdbcTemplate
JdbcTemplate 概述 JdbcTemplate是Spring提供的一个模板类,它是对jdbc的封装.用于支持持久层的操作.具有简单,方便等特点. pom.xml <!--依赖版本--& ...
- CSS——NO.8(代码简写)
*/ * Copyright (c) 2016,烟台大学计算机与控制工程学院 * All rights reserved. * 文件名:text.cpp * 作者:常轩 * 微信公众号:Worldhe ...
- PAT-进制转换
3.5-进制转换 对于一个P进制的数,如果要转换为Q进制的数,需要分为两步: ①将P进制数x转换为十进制数y 对于一个十进制数y=d1d2···dn,可以将其写为: y = d1 * 10n-1 + ...
- 时间序列数据库(TSDB)初识与选择(InfluxDB、OpenTSDB、Druid、Elasticsearch对比)
背景 这两年互联网行业掀着一股新风,总是听着各种高大上的新名词.大数据.人工智能.物联网.机器学习.商业智能.智能预警啊等等. 以前的系统,做数据可视化,信息管理,流程控制.现在业务已经不仅仅满足于这 ...
- 从头认识js-函数表达式
定义函数的方式有两种: 1.函数声明(特征:函数声明提升,在执行代码之前会先读取函数声明,这就意味着可以把函数声明放在调用它的语句之后) 2.函数表达式(函数表达式与其他表达式一样,使用之前必须先声明 ...