换根 DP

news/2025/1/23 17:44:59/文章来源:https://www.cnblogs.com/ronchen/p/18536275

树形 DP 中的换根 DP 问题又被称为二次扫描,通常需要求以每个点为根时某个式子的答案。

这一类问题通常需要遍历两次树,第一次遍历先求出以某个点(如 \(1\))为根时的答案,在第二次遍历时考虑由根为 \(u\) 转化为根为 \(v\) 时答案的变化(换根)。这个变化往往分为两部分,\(v\) 子树外的点到 \(v\) 相比于到 \(u\) 会增加一条边,而 \(v\) 子树内的点到 \(v\) 相比于到 \(u\) 会减少一条边。

所以往往在第一次遍历时可以顺带求出一些子树信息,利用这些信息辅助第二次遍历时的换根操作。

经典例题:求对于每个点而言,其他点到这个点的距离之和。

例题:P3478 [POI2008] STA-Station

给定一棵 \(n\) 个节点的树,求出一个节点,使得以这个节点为根时,所有节点的深度之和最大。
数据范围:\(n \le 10^6\)

分析:随便选择一个节点 \(u\) 作为根节点,遍历整棵树,则得到了以 \(u\) 为根节点时的深度之和。

\(dp_u\) 表示以 \(u\) 为根时,所有节点的深度之和。设 \(v\) 为当前节点的某个子节点,考虑“换根”,即以 \(u\) 为根转移到以 \(v\) 为根,显然在换根的过程中,以 \(v\) 为根会导致每个节点的深度都产生改变。具体表现为:

  • 所有在 \(v\) 的子树上的节点深度都减少了一,那么总深度和就减少了 \(sz_v\),这里用 \(sz_i\) 表示以 \(i\) 为根的子树中的节点个数。
  • 所有不在 \(v\) 的子树上的节点深度都增加了一,那么总深度和就增加了 \(n - sz_v\)

根据这两个条件就可以推出状态转移方程:\(dp_v = dp_u + n - 2 \times sz_v\),因此可以在第一次遍历时顺便计算一下 \(sz\),第二次遍历时用状态转移方程计算出最终的答案。

参考代码
#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
typedef long long LL;
const int N = 1000005;
vector<int> tree[N];
int sz[N], res, n;
LL ans;
void dfs(int cur, int fa, int depth) {ans += depth;sz[cur] = 1;for (int to : tree[cur]) {if (to == fa) continue;dfs(to, cur, depth + 1);sz[cur] += sz[to];}
}
void solve(int cur, int fa, LL sum) {for (int to : tree[cur]) {if (to == fa) continue;LL tmp = sum + n - 2 * sz[to];if (tmp > ans) {ans = tmp; res = to;}solve(to, cur, tmp);}
}
int main()
{scanf("%d", &n);for (int i = 1; i < n; i++) {int u, v; scanf("%d%d", &u, &v);tree[u].push_back(v); tree[v].push_back(u);}dfs(1, 0, 1);res = 1;solve(1, 0, ans);printf("%d\n", res);return 0;
}

习题:P2986 [USACO10MAR] Great Cow Gathering G

解题思路

与上一题类似,只不过这题换根时的变化量是点权和(即牛棚中奶牛的数量)的变化乘以边权。

参考代码
#include <cstdio>
#include <vector>
using namespace std;
typedef long long LL;
const int N = 100005;
int n, c[N];
LL ans;
struct Edge {int to, l;
};
vector<Edge> tree[N];
void dfs(int cur, int fa, int depth) {ans += 1ll * depth * c[cur];for (Edge e : tree[cur]) {if (e.to == fa) continue;dfs(e.to, cur, depth + e.l);c[cur] += c[e.to];}
}
void solve(int cur, int fa, LL sum) {for (Edge e : tree[cur]) {if (e.to == fa) continue;LL tmp = sum + 1ll * (c[1] - 2 * c[e.to]) * e.l;ans = min(ans, tmp);solve(e.to, cur, tmp);}
}
int main()
{scanf("%d", &n);for (int i = 1; i <= n; i++) scanf("%d", &c[i]);for (int i = 1; i < n; i++) {int a, b, l; scanf("%d%d%d", &a, &b, &l);tree[a].push_back({b, l}); tree[b].push_back({a, l});}dfs(1, 0, 0);solve(1, 0, ans);printf("%lld\n", ans);return 0;
}

例题:P3047 [USACO12FEB] Nearby Cows G

分析:可以先对树做一次遍历得到每个节点对应的子树下距离子树根节点距离 \(0 \sim x\) 之间的点权和,

然后考虑每个点距离 \(k\) 之内的点权和。子树下的点权和在第一次遍历时已经计算完成,因此还需要计算的是在该点子树外的距离 \(k\) 以内的部分,而这个部分可以通过对该点上方最多 \(k\) 个祖先节点的处理,如下图所示。

image

参考代码
#include <cstdio>
#include <vector>
using std::vector;
const int N = 100005;
const int K = 25;
vector<int> tree[N];
int n, k, sum[N][K], c[N], ans[N];
void dfs(int u, int fa) {for (int v : tree[u]) {if (v == fa) continue;dfs(v, u);for (int i = 1; i <= k; i++) {sum[u][i] += sum[v][i - 1];}}sum[u][0] = c[u];
}
void calc(int u, int fa, vector<int> pre) {int dis = pre.size();pre.push_back(u);// u的子树内的距离范围内的点权和ans[u] = sum[u][k];// 计算u的子树外的距离范围内的点权和for (int i = 0; i + 1 < pre.size(); i++) {int cur = pre[i], nxt = pre[i + 1];// 对于边cur->nxtans[u] += sum[cur][k - dis]; // 加上cur子树下的距离内点权和if (k - dis > 0) ans[u] -= sum[nxt][k - dis - 1]; // 减去nxt子树下刚刚被重复计算的部分dis--;}vector<int> path;if (pre.size() == k + 1) {// pre[0]即将超出下面的点的距离范围k,要被淘汰for (int i = 1; i < pre.size(); i++) path.push_back(pre[i]);} else path = pre;for (int v : tree[u]) {if (v == fa) continue;calc(v, u, path);}
}
int main()
{scanf("%d%d", &n, &k);for (int i = 1; i < n; i++) {int u, v; scanf("%d%d", &u, &v);tree[u].push_back(v); tree[v].push_back(u);}for (int i = 1; i <= n; i++) scanf("%d", &c[i]);dfs(1, 0);// 生成前缀和for (int i = 1; i <= n; i++) {for (int j = 1; j <= k; j++) sum[i][j] += sum[i][j - 1];}vector<int> tmp;calc(1, 0, tmp);for (int i = 1; i <= n; i++) printf("%d\n", ans[i]);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/829146.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

学习笔记(三十):ArkUi-UIContext.getPromptAction(弹窗)

概述: 基于promptAction弹窗演进而来,支持全局自定义弹窗,不依赖UI组件,依赖UIContext, 支持在非页面文件中使用,弹窗内容支持动态修改,支持自定义弹窗圆角半径、大小和位置, 适合在与页面解耦的全局弹窗、自定义弹窗显示和退出动画等场景下使用。 注意: 需先使用UICo…

MudBlazor:基于Material Design风格开源且强大的Blazor组件库

项目介绍 MudBlazor是一个基于Material Design风格开源、免费(MIT License)、功能强大的Blazor组件框架,注重易用性和清晰的结构。它非常适合想要快速构建Web应用程序的 .NET 开发人员,无需费力地处理 CSS 和 JavaScript。由于MudBlazor完全使用C#编写,因此你可以自由地调…

读数据工程之道:设计和构建健壮的数据系统32序列化和云网络

序列化和云网络1. 序列化 1.1. 仅仅通过从CSV转换到Parquet序列化,任务性能就提高了上百倍 1.2. 基于行的序列化1.2.1. 基于行的序列化是按行来组织数据1.2.2. 对于那些半结构化的数据(支持嵌套和模式变化的数据对象)​,基于行的序列化需要将每个对象作为一个单元来存储1.2…

入门龙芯旧世界汇编指令

我是龙芯汇编指令新手,本文是我学习龙芯汇编的笔记我借到了一台宝贵的龙芯 3A6000 设备,我期望在这台设备上面学习龙芯汇编指令。这台设备上的是龙芯旧世界的麒麟系统,由于这台设备很宝贵,我不能随意玩。为了防止弄坏设备,我将在此设备上面搭建 docker 环境,进入到 docke…

促进通用跨域检索中广义知识的模拟

促进通用跨域检索中广义知识的模拟ProS:促进通用跨域检索中广义知识的模拟通用跨域检索(UCDR)的目标是在广义测试场景中实现稳健的性能,其中数据在训练过程中可能属于严格未知的域和类别。最近,具有快速调整的预训练模型显示出很强的泛化能力,并在各种下游任务中取得了显著…

ParamISP:使用相机参数学习正向和反向ISP

ParamISP:使用相机参数学习正向和反向ISPRAW图像很少被共享,主要是因为与相机ISP获得的sRGB图像相比,RAW图像的数据量过大。最近已经证明,学习相机ISP的正向和反向过程,可以对输入的sRGB图像进行具有物理意义的RAW级图像处理。然而,现有的基于学习的ISP方法,无法处理ISP…

thinkphp console 命令行打印错误调用堆栈

在think\Console源文件里找到 run() 方法,加上内容: $output->error($e->getTraceAsString()); 然后当执行命令报错的时候就会有详细的错误信息,方便排查具体是哪行引起的问题!本文来自博客园,作者:imzhi,转载请注明原文链接:https://www.cnblogs.com/imzhi/p/18…

信道的极限容量

我们可以简单地将带通信道理解为无线传输信道,低通信道理解为有线传输信道,记忆公式时应该记住乘2的那个调制速度就是波特率

JD 商品詳情頁解析

https://item.jd.com/100036218692.html 以这个商品链接为例,分析详情图接口抓包拿到接口入参出参构建代码headers = {cookie:"",accept: application/json, text/javascript, */*; q=0.01,accept-language: zh-CN,zh;q=0.9,origin: https://item.jd.com,priority: …

Vmware虚拟机下载安装使用教程(17.5.2最新版非常详细)

VMware,自1998年成立以来,便以其革命性的虚拟化技术引领行业。这家公司专注于提供软件和服务,支持云计算和虚拟化解决方案,使得一台物理服务器能够托管多个独立的虚拟机,每个虚拟机都能独立运行不同的操作系统和应用。VMware的技术不仅优化了硬件资源的使用效率,还简化了…

Vmware虚拟机下载安装使用教程(2024最新版非常详细)

VMware,自1998年成立以来,便以其革命性的虚拟化技术引领行业。这家公司专注于提供软件和服务,支持云计算和虚拟化解决方案,使得一台物理服务器能够托管多个独立的虚拟机,每个虚拟机都能独立运行不同的操作系统和应用。VMware的技术不仅优化了硬件资源的使用效率,还简化了…