换根 DP 学习笔记
前言
没脑子选手什么都不会。
正文
先来写一下换根 DP 的特点或应用方面:
- 不同的点作为树的根节点,答案不一样。
- 求解答案时要求出每一个节点的信息。
- 无法通过一次搜索完成答案的求解,因为一次搜索只能得到一个节点的答案。
下面来看一个例子:
给定一个 \(n\) 个点的无根树,问以树上哪个节点为根时,其所有节点的深度和最大。
一个显然的做法:枚举根节点然后 \(O(n)\) 暴力,复杂度 \(O(n^2)\) 。能过我 CS
所以我们考虑换根 DP 。
- 先以 \(1\) 节点为根节点算出每个点的深度 \(deep_i\) 和每个点为根的子树大小 \(siz_i\)
- 那么此时我们就知道了 \(1\) 为根节点时的答案 \(ans_1=\sum deep_i\) 。
- 接下来我们来看第二遍 \(\text{dfs}\) ,考虑如何由 \(ans_1\) 转换到 \(ans_i\) 。
- 因为还是从 \(1\) 节点开始向下遍历,所以默认 \(t\) 是 \(u\) 的孩子节点。
- 很显然,转移的时候把树分成两个块:1.本来就是 \(t\) 的子树 2. 原来不是 \(t\) 的子树。
- 先来考虑原来就是 \(t\) 子树的情况:每一个节点的值都要减一,所以
ans -= siz[t] - 在来考虑另一种情况:每个节点显然答案加一,所以
ans += n - siz[t]
\]
Code
#include <cstdio>
#include <vector>
#include <iostream>
#include <algorithm>
#define file(a) freopen(a".in", "r", stdin), freopen(a".out", "w", stdout)
#define Enter putchar('\n')
#define quad putchar(' ')
#define int long long
const int N = 1000005;
int n, deep[N], siz[N], f[N];
std::vector <int> dis[N];
inline void dfs1(int now, int father) {
deep[now] = deep[father] + 1;
siz[now] = 1;
for (int t : dis[now]) {
if (t == father) continue;
dfs1(t, now);
siz[now] += siz[t];
}
}
inline void dfs2(int now, int father) {
for (int t : dis[now]) {
if (t == father) continue;
f[t] = f[now] + n - 2 * siz[t];
dfs2(t, now);
}
}
signed main(void) {
// file("P3478");
std::ios::sync_with_stdio(false);
std::cin.tie(0);
std::cout.tie(0);
std::cin >> n;
for (int i = 1, x, y; i < n; i++) {
std::cin >> x >> y;
dis[x].emplace_back(y);
dis[y].emplace_back(x);
}
dfs1(1, 0);
for (int i = 1; i <= n; i++)
f[1] += deep[i];
dfs2(1, 0);
int ans = 0, out;
for (int i = 1; i <= n; i++) {
if (f[i] > ans) {
ans = f[i];
out = i;
}
}
std::cout << out << std::endl;
return 0;
}
所以我们可以发现:
换根 DP 一般都是先选择任意一个点为根节点预处理出一些有用的信息。
然后第二遍 dfs 时再根据已知的答案推出其他节点的答案。
换根 DP 学习笔记的更多相关文章
- [算法学习] 换根dp
换根dp 一般来说,我们做题的树都是默认 \(1\) 为根的.但是有些题目需要计算以每个节点为根时的内容. 朴素的暴力:以每个点 \(u\) 作为 \(root\) 暴力dfs下去,复杂度\(O(n^ ...
- 树形DP 学习笔记
树形DP学习笔记 ps: 本文内容与蓝书一致 树的重心 概念: 一颗树中的一个节点其最大子树的节点树最小 解法:对与每个节点求他儿子的\(size\) ,上方子树的节点个数为\(n-size_u\) ...
- [BZOJ4379][POI2015]Modernizacja autostrady[树的直径+换根dp]
题意 给定一棵 \(n\) 个节点的树,可以断掉一条边再连接任意两个点,询问新构成的树的直径的最小和最大值. \(n\leq 5\times 10^5\) . 分析 记断掉一条边之后两棵树的直径为 \ ...
- 数位DP学习笔记
数位DP学习笔记 什么是数位DP? 数位DP比较经典的题目是在数字Li和Ri之间求有多少个满足X性质的数,显然对于所有的题目都可以这样得到一些暴力的分数 我们称之为朴素算法: for(int i=l_ ...
- 2018.10.15 NOIP训练 水流成河(换根dp)
传送门 换根dp入门题. 貌似李煜东的书上讲过? 不记得了. 先推出以1为根时的答案. 然后考虑向儿子转移. 我们记f[p]f[p]f[p]表示原树中以ppp为根的子树的答案. g[p]g[p]g[p ...
- DP学习笔记
DP学习笔记 可是记下来有什么用呢?我又不会 笨蛋你以后就会了 完全背包问题 先理解初始的DP方程: void solve() { for(int i=0;i<;i++) for(int j=0 ...
- 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市
P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...
- 小奇的仓库:换根dp
一道很好的换根dp题.考场上现场yy十分愉快 给定树,求每个点的到其它所有点的距离异或上m之后的值,n=100000,m<=16 只能线性复杂度求解,m又小得奇怪.或者带一个log像kx一样打一 ...
- 国家集训队 Crash 的文明世界(第二类斯特林数+换根dp)
题意 题目链接:https://www.luogu.org/problem/P4827 给定一棵 \(n\) 个节点的树和一个常数 \(k\) ,对于树上的每一个节点 \(i\) ,求出 \( ...
随机推荐
- Sliding Window - 题解【单调队列】
题面: An array of size n ≤ 106 is given to you. There is a sliding window of size k which is moving fr ...
- 【FAQ】HMS Core推送服务与本地创建通知消息如何相互覆盖?
我们知道,单独使用HMS Core推送服务或本地创建通知消息,都可以实现通知消息的覆盖,方式分别为: 1.本地创建通知消息(简称本地通知消息) 通过notificationManager.notify ...
- C++中 指针的指针是什么?指针的引用又是什么?你可能需要看看这篇文章
关于变量的定义 我们都知道变量的定义包括一个基本数据类型(base type)和一组声明符,在同一条定义语句中,输入基本数据类型不同,但是声明符的形式却可以不同. //如:i是一个int的整数,a是一 ...
- 如何在Web前端实现CAD图文字全文搜索功能之技术分享
现状 在CAD看图过程中我们经常会需要用到查找文字的功能,在AutoCAD软件查找一个文字时,可以通过打开左下角输入命令find,输入查找的文字,然后设置查找范围,就可以搜索到需要查询的文字.但在We ...
- Http GET 请求参数中文乱码
两种解决方式 第1种:代码里转换 String name = request.getParamter("name"); String nameUtf8 = new String(n ...
- [源码解析] TensorFlow 分布式之 MirroredStrategy 分发计算
[源码解析] TensorFlow 分布式之 MirroredStrategy 分发计算 目录 [源码解析] TensorFlow 分布式之 MirroredStrategy 分发计算 0x1. 运行 ...
- 【ACM程序设计】求最小生成树 Kuskual算法
Kuskual算法 流程 1 将图G看做一个森林,每个顶点为一棵独立的树 2 将所有的边加入集合S,即一开始S = E( 并查集) 3 从S中拿出一条最短的边(u,v),如果(u,v)不在同一棵树内, ...
- 机器学习-学习笔记(一) --> (假设空间 & 版本空间)及 归纳偏好
机器学习 一.机器学习概念 啥是机器学习 机器学习:假设用P来评估计算机程序在某任务类T上的性能,若一个程序通过利用经验E在T中任务上获得了性能改善,则关于T和P,该程序对E进行了学习 通俗讲:通过计 ...
- Python写安全小工具-TCP全连接端口扫描器
通过端口扫描我们可以知道目标主机都开放了哪些服务,下面通过TCP connect来实现一个TCP全连接端口扫描器. 一个简单的端口扫描器 #!/usr/bin/python3 # -*- coding ...
- 联发科 (MTK) sensor bring up
MT6768平台 1.添加驱动文件 2.添加硬件配置支持 3.添加硬件配置 4.添加编译配置 5.分配空间(非必要,当代码量超过当前空间大小时将会报错,根据报错log改大小即可.) 6.兼容配置 7. ...