虚树(Virtual Tree)学习笔记

虚树(Virtual Tree)学习笔记


一道题目(BZOJ-2286消耗战)

Description
在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达。现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军胜利在望。已知在其他k个岛屿上有丰富能源,为了防止敌军获取能源,我军的任务是炸毁一些桥梁,使得敌军不能到达任何能源丰富的岛屿。由于不同桥梁的材质和结构不同,所以炸毁不同的桥梁有不同的代价,我军希望在满足目标的同时使得总代价最小。
侦查部门还发现,敌军有一台神秘机器。即使我军切断所有能源之后,他们也可以用那台机器。机器产生的效果不仅仅会修复所有我军炸毁的桥梁,而且会重新随机资源分布(但可以保证的是,资源不会分布到1号岛屿上)。不过侦查部门还发现了这台机器只能够使用m次,所以我们只需要把每次任务完成即可。
Input
第一行一个整数n,代表岛屿数量。
接下来n-1行,每行三个整数u,v,w,代表u号岛屿和v号岛屿由一条代价为c的桥梁直接相连,保证1<=u,v<=n且1<=c<=100000。
第n+1行,一个整数m,代表敌方机器能使用的次数。
接下来m行,每行一个整数ki,代表第i次后,有ki个岛屿资源丰富,接下来k个整数h1,h2,…hk,表示资源丰富岛屿的编号。
Output
输出有m行,分别代表每次任务的最小代价。


树的点数少

如果树的点数少,可以用\(dp[u]\)表示子树的所有关键点都不连通的最小代价,如果子节点\(v\)是关键点,则一定断开\(uv\)这条边,否则选择是断开\(uv\),还是使用子树内的边来阻断,转移即可,\(O(nq)\)


虚树 Virtual Tree

  1. 不难发现其中很多点是无用的,对我们有用的点其实只有所有的关键点和他们之间的\(lca\),虚树就是在不改变祖先与后代关系的情况下,保留了一些有用点,把一整颗树的信息进行压缩,建出一棵小树。

  2. 我们可以通过将关键点按\(dfs\)序排序,求出一些相邻点的\(lca\),将他们加入虚树中,同时保持祖先与后代的关系,这里可以看出询问\(n\)个关键点时,虚树的节点数N小于等于\(2n-1\)

  3. 使用单调栈建虚树,为了方便我们把\(1\)加入虚树中。用单调栈来维护一条虚树上的链,也就是一个栈里相邻的两个节点在虚树上也是相邻的,而且栈是从底部到栈首单调\(dfs\)序递增的。

  4. 太懒了。。算法细节还是看这里

  5. 对于本题,建出虚树后我们还要维护一些被压缩路径的最小值,可以倍增维护

#include <bits/stdc++.h>
typedef long long ll;
const int N = 250010;
using namespace std;
int n, q;
struct edge { int e, w, nxt; } E[N << 1];
int cc, h[N];
void add(int u,int v,int w) {E[cc].e = v; E[cc].w = w; E[cc].nxt = h[u]; h[u] = cc; ++cc;
}
int dfn[N], id, fa[N][20], dep[N], mn[N][20];
void dfs(int u, int pre) {dfn[u] = ++ id; fa[u][0] = pre; dep[u] = dep[pre] + 1;for(int i = 1; i <= 19; ++i) {fa[u][i] = fa[fa[u][i-1]][i-1];mn[u][i] = min(mn[u][i-1], mn[fa[u][i-1]][i-1]);}for(int i = h[u]; ~i; i = E[i].nxt) {int v = E[i].e, w = E[i].w;if(v == pre) continue;mn[v][0] = w; dfs(v, u);}
}
int lca(int u, int v) {if(dep[u] > dep[v]) swap(u, v);for(int i = 19; i >= 0; --i) if(dep[fa[v][i]] >= dep[u]) v = fa[v][i];if(u == v) return v;for(int i = 19; i >= 0; --i) if(fa[u][i] != fa[v][i]) u = fa[u][i], v = fa[v][i];return fa[u][0];
}
int calmn(int u, int v) {int ans = INT_MAX;if(dep[u] > dep[v]) swap(u, v);for(int i = 19; i >= 0; --i) if(dep[fa[v][i]] >= dep[u]) ans = min(ans, mn[v][i]),  v = fa[v][i];return ans;
}
bool cmp(int u, int v) { return dfn[u] < dfn[v]; }
int a[N], mk[N], stc[N], top;
void build(int a[], int m) {sort(a+1, a+1+m, cmp); stc[top = 1] = 1; cc = 0; h[1] = -1;for(int p, i = 1; i <= m; ++i) {if(a[i] != 1) {p = lca(stc[top], a[i]);if(p != stc[top]) {while(dfn[p] < dfn[stc[top-1]]) {int w = calmn(stc[top-1],stc[top]);add(stc[top-1],stc[top],w); -- top;}if(dfn[p] > dfn[stc[top-1]]) {int w = calmn(p,stc[top]); h[p] = -1;add(p, stc[top], w), stc[top] = p;}else {int w = calmn(p,stc[top]);add(p, stc[top--], w);}}h[a[i]] = -1, stc[++top] = a[i];}}for(int w, i = 1; i < top; ++i) {w = calmn(stc[i], stc[i+1]);add(stc[i],stc[i+1],w);}
}
ll dp[N];
void Dp(int u) {dp[u] = 0;for(int i = h[u]; ~i; i = E[i].nxt) {Dp(E[i].e);if(mk[E[i].e]) dp[u] += E[i].w;else dp[u] += min(1ll*E[i].w, dp[E[i].e]);}
}
int main() {scanf("%d",&n); memset(h,-1,sizeof(h));for(int u, v, w, i = 1; i < n; ++i) {scanf("%d%d%d",&u,&v,&w);add(u,v,w), add(v,u,w);}   dfs(1,0);scanf("%d",&q);while(q--) { int m;scanf("%d",&m);for(int i = 1; i <= m; ++i) scanf("%d",&a[i]), mk[a[i]] = 1;build(a, m); Dp(1);printf("%lld\n", dp[1]);for(int i = 1; i <= m; ++i) mk[a[i]] = 0;}
}

转载于:https://www.cnblogs.com/RRRR-wys/p/10310371.html

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

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

相关文章

利用Asp.Net Core的MiddleWare思想处理复杂业务流程

最近利用Asp.Net Core 的MiddleWare思想对公司的古老代码进行重构&#xff0c;在这里把我的设计思路分享出来&#xff0c;希望对大家处理复杂的流程业务能有所帮助。背景一个流程初始化接口&#xff0c;接口中根据传入的流程类型&#xff0c;需要做一些不同的工作。1.有的工作是…

P4284-[SHOI2014]概率充电器【树形dp,换根法,数学期望】

正题 题目链接:https://www.luogu.com.cn/problem/P4284 题目大意 nnn个点的一棵树&#xff0c;每个点有pip_ipi​概率通电&#xff0c;每个边有一定概率可以导电。求期望有电的节点个数。 解题思路 发现考虑通电概率很难&#xff0c;所以我们可以考虑不通电的概率。 如果只…

【每日一题】4月7日题目精讲 树

树 时间限制&#xff1a;C/C 1秒&#xff0c;其他语言2秒 空间限制&#xff1a;C/C 131072K 其他语言262144K 64bit IO Format:%lld 题目描述 shy有一颗树&#xff0c;树有n个结点。有k种不同颜色的染料给树染色。一个染色方案是合法的&#xff0c;当且仅当对于所有相同颜色的点…

【贪心】Radar Installation(poj 1328)

Radar Installation poj 1328 题目大意&#xff1a; 在平面直角坐标系的一二象限上有n个小岛&#xff0c;现在让你在x坐标上布置雷达&#xff0c;每个雷达可以侦测以它为原心&#xff0c;半径为m的圆内的所有小岛&#xff0c;现在问侦测完这n个小岛最少要多少个雷达 输入样…

HDU5765 Bonds (高维前缀和)

HDU5765 Bonds (高维前缀和) 题意&#xff1a;\(n(n<20)\)个点\(m\)条边无向图&#xff0c;求每条边出现在多少个\(Bond\)里。一个图的\(cut\)指&#xff0c;对于一个图\(G\)的边集的某个子集\(E\)&#xff0c;如果删除\(E\)中的所有边&#xff0c;原图不连通。一个图的\(Bo…

F# 4.5提供Spans、Match!等特性

F# 4.5预览版现已发布&#xff0c;其中提供了一系列新特性&#xff0c;包括对.NET Core 2.1的新原生类型Span<T>的支持、新关键字Match!等。类型Span意在实现底层代码指针操作的安全性和可预测性&#xff0c;这可使得很多情况下不必再分配内存&#xff0c;进而改进了内存…

P4172-[WC2006]水管局长【LCT,最小生成树】

正题 题目链接:https://www.luogu.com.cn/problem/P4172 题目大意 nnn个点的一张图&#xff0c;有两个操作 询问两个点之间的一条路径使得最长边最短删除一条边 解题思路 显然这条边一定是在最小生成树上的&#xff0c;所以我们需要维护支持删边的最小生成树。 显然LCTLCTL…

牛客OI周赛15-普及组

1. 咪咪游戏 bool f1,f2,f; f1用于记录m f2用于记录q f用于判断串的好坏。1为坏&#xff0c;0为好 注意几种情况&#xff1a; mqm mmq 具体看代码&#xff1a; #include<bits/stdc.h> using namespace std; int main() {int n;cin>>n;string s;while(n--){bool f10…

【单调栈】向右看齐(luogu 2947)

向右看齐 luogu 2947 题目大意&#xff1a; 有n头奶牛&#xff0c;它们各有各的高度&#xff0c;现在要求出每一头奶牛右边第一头比他高的奶牛是那一头 输入样例 6 3 2 6 1 1 2 输出样例 3 3 0 6 6 0 输入说明 6 头奶牛的身高分别为 3, 2, 6, 1, 1, 2. 输出…

Codeforces 786B Legacy (线段树优化建图)

Codeforces 786B Legacy (线段树优化建图) 题意&#xff1a;\(n\)个点&#xff0c;有\(3\)种连边操作&#xff1a;1.将\(u\)指向\(v\)&#xff1b;2.将\(v\)指向编号在区间\([l,r]\)的点&#xff1b;3.将\([l,r]\)中的所有点指向\(v\) 做法&#xff1a;线段树优化建图。拓展一些…

Abp + Grpc 如何实现用户会话状态传递

0.背景在实际项目当中&#xff0c;我采用的是 Abp 框架&#xff0c;但是 Abp 框架官方并没有针对 Grpc 进行模块封装。基于此我结合 Abp 与 MagicOnion 封装了一个 Abp.Grpc 模块&#xff0c;它包括服务端和调用端两部分的包。通过这两个包&#xff0c;你可以很方便地在 Abp 框…

P4197-Peaks【Kruskal重构树,主席树】

正题 题目链接:https://www.luogu.com.cn/problem/P4197 题目大意 nnn个点的一张无向图&#xff0c;每个点有一个hih_ihi​&#xff0c;边有权值。 qqq次询问从vvv出发不走权值超过xxx的路径能到达的第kkk大hih_ihi​是多少。 解题思路 构一颗KruskalKruskalKruskal重构树之后…

多项式相关操作学习笔记

多项式相关操作学习笔记 标签&#xff1a; 多项式 说在前边 记录一下相关的多项式操作&#xff0c;顺便存个模板。&#xff08;多点求值之后的部分&#xff0c;有点写不动了。。。留坑留坑 多项式 定义 给定一个环\(R\)(\(R\)通常是交换环&#xff0c;可以是有理数、实数或者复…

恢复数列

题目链接 比赛链接 时间限制&#xff1a;C/C 1秒&#xff0c;其他语言2秒 空间限制&#xff1a;C/C 262144K&#xff0c;其他语言524288K Special Judge,64bit IO Format: %lld 题目描述 小y的数学作业不小心被泼上了墨水。有道题看不清了&#xff0c;现在他想请你帮他恢复这道…

【单调栈】Largest Rectangle in a Histogram(luogu-SP1805/poj 2559)

Largest Rectangle in a Histogram luogu-SP1805 poj 2559 题目大意&#xff1a; 有n个并排的矩阵&#xff0c;高度为aia_iai​&#xff0c;宽度为1&#xff0c;现在让求包含于这些矩阵的并集内部的最大矩阵的面积 输入样例 7 2 1 4 5 1 3 3 4 1000 1000 1000 1000 0输出…

【翻译】asp.net core中使用MediatR

这篇文章来自&#xff1a;https://ardalis.com/using-mediatr-in-aspnet-core-apps本文作为翻译&#xff0c;有一些单词翻译成中文可能会有一些误解&#xff08;对于读者&#xff09;或者错误&#xff08;对于作者&#xff09;的地方&#xff0c;所以在文章中你可以看到一些单词…

P2056-[ZJOI2007]捉迷藏【点分树,堆】

正题 题目链接:https://www.luogu.com.cn/problem/P2056 题目大意 nnn个点的一棵树&#xff0c;开始全是黑点&#xff0c;有操作 取反一个点的颜色求最远的黑点之间的距离 解题思路 根据点分治每个点和分散开来的重心连边&#xff0c;然后每个点往上只会有logloglog层节点。…

数论杂谈(欧拉定理与费马小定理结论与应用)

文章目录欧拉定理&#xff1a;欧拉定理性质&#xff1a;扩展欧拉定理&#xff1a;费马小定理&#xff1a;指数循环节费马大定理逆元&#xff1a;例题原根定义&#xff1a;原根存在条件例题快速幂代码矩阵快速幂原理&#xff1a;代码&#xff1a;欧拉定理&#xff1a; aφ(n)≡…

【队列】Team Queue(luogu-UVA540/poj 2259)

Team Queue luogu-UVA540 poj 2259 题目大意&#xff1a; 有n个小组的人要排队&#xff0c;每个小组中有若干个人&#xff0c;当一个人入队时&#xff0c;如果队中有自己小组的人就跟在此人后面&#xff0c;否则站在队尾&#xff0c;现在给出一些操作&#xff0c;让你去操作…

ASP.NET Core MVC with EF Core-迁移

当你开发一个新的应用程序的时候&#xff0c;你的模型频繁的变化&#xff0c;而每一次的数据模型的改变&#xff0c;将使它与数据库不同步。你通过配置EF Core&#xff0c;使得数据库不存在时创建数据库。每一次改变数据模型&#xff08;增删改 实体类或者改变DbContextClass),…