luoguP2495 [SDOI2011]消耗战
https://www.luogu.org/problemnew/show/P2495
Dp 方程很显然
设 Dp[u] 表示——使 u 不与其子树中任意一个关键点联通的最小代价
设 w[a, b] 表示 a 与 b 之间的边的权值。
- 若 son[i] 不是关键点,Dp[u] = Dp[u] + min{Dp[son[i]],w[u][son[i]]}
- 若 son[i] 是关键点,Dp[u] = Dp[u] + w[u][son[i]]
但这样复杂度很显然是不对的,所以我们考虑虚树
什么,你还不会虚树?那就去跟 zzq 学吧 https://www.cnblogs.com/zzqsblog/p/5560645.html
我们发现 k 的总和与 n 同级,所以用虚树优化这个 Dp,建出虚树,在虚树上 Dp 即可
#include <bits/stdc++.h>
#define X first
#define Y second
#define mp make_pair
using namespace std;
typedef long long ll;
const int N = 250000 + 5, LG2 = 18;
vector < pair <int, int> > G[N], G2[N];
int pre[N][LG2 + 1], dep[N], mx[N][LG2 + 1], id[N], dfn;
int n, m, k, h[N], sta[N], len, MX;
ll f[N];
bool book[N];
void init(int u, int fa) {
pre[u][0] = fa; dep[u] = dep[fa] + 1; id[u] = ++dfn;
for(int i = 1; i <= LG2; i++) {
pre[u][i] = pre[pre[u][i - 1]][i - 1];
mx[u][i] = min(mx[u][i - 1], mx[pre[u][i - 1]][i - 1]);
}
for(vector < pair <int, int> > :: iterator it = G[u].begin(); it != G[u].end(); it++) {
int v = it -> X; if(v != fa) mx[v][0] = it -> Y, init(v, u);
}
}
int LCA(int x, int y) {
MX = INT_MAX;
if(dep[x] > dep[y]) swap(x, y);
for(int i = LG2; i >= 0; i--)
if(dep[pre[y][i]] >= dep[x])
MX = min(MX, mx[y][i]), y = pre[y][i];
if(x == y) return x;
for(int i = LG2; i >= 0; i--)
if(pre[x][i] != pre[y][i]) {
MX = min(MX, min(mx[x][i], mx[y][i]));
x = pre[x][i], y = pre[y][i];
}
return pre[x][0];
}
bool cmp(int x, int y) {return id[x] < id[y];}
void DP(int u) {
f[u] = 0;
for(vector < pair <int, int> > :: iterator it = G2[u].begin(); it != G2[u].end(); it++) {
int v = it -> X; DP(v);
if(book[v]) f[u] += it -> Y;
else f[u] += min(f[v], (ll)it -> Y);
}
}
int main() {
cin >> n;
for(int i = 1; i < n; i++) {
int a, b, c;
scanf("%d %d %d", &a, &b, &c);
G[a].push_back( mp(b, c) );
G[b].push_back( mp(a, c) );
}
init(1, 0); cin >> m;
while(m--) {
scanf("%d", &k);
for(int i = 1; i <= k; i++) {
scanf("%d", &h[i]);
book[h[i]] = 1;
}
sort(h + 1, h + k + 1, cmp);
sta[len = 1] = 1; G2[1].clear();
for(int i = 1; i <= k; i++) {
if(h[i] == 1) continue;
int lca = LCA(h[i], sta[len]);
if(lca != sta[len]) {
while(id[lca] < id[sta[len - 1]]) {
LCA(sta[len - 1], sta[len]);
G2[sta[len - 1]].push_back( mp(sta[len], MX) );
len--;
}
if(id[lca] > id[sta[len - 1]]) {
G2[lca].clear();
LCA(lca, sta[len]);
G2[lca].push_back( mp(sta[len], MX) );
sta[len] = lca;
} else LCA(lca, sta[len]), G2[lca].push_back( mp(sta[len], MX) ), len--;
}
G2[h[i]].clear(); sta[++len] = h[i];
}
for(int i = 1; i < len; i++) LCA(sta[i], sta[i + 1]), G2[sta[i]].push_back( mp(sta[i + 1], MX) );
DP(1); printf("%lld\n", f[1]);
for(int i = 1; i <= k; i++) book[h[i]] = 0;
}
return 0;
}
luoguP2495 [SDOI2011]消耗战的更多相关文章
- [luoguP2495] [SDOI2011]消耗战(DP + 虚树)
传送门 明显虚树. 别的题解里都是这样说的. 先不考虑虚树,假设只有一组询问,该如何dp? f[u]表示把子树u中所有的有资源的节点都切掉的最优解 如果节点u需要切掉了话,$f[u]=val[u]$ ...
- BZOJ 2286: [Sdoi2011]消耗战
2286: [Sdoi2011消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2082 Solved: 736[Submit][Status] ...
- bzoj 2286: [Sdoi2011]消耗战 虚树+树dp
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 在一 ...
- bzoj千题计划254:bzoj2286: [Sdoi2011]消耗战
http://www.lydsy.com/JudgeOnline/problem.php?id=2286 虚树上树形DP #include<cmath> #include<cstdi ...
- 【LG2495】[SDOI2011]消耗战
[LG2495][SDOI2011]消耗战 题面 洛谷 题解 参考博客 题意 给你\(n\)个点的一棵树 \(m\)个询问,每个询问给出\(k\)个点 求将这\(k\)个点与\(1\)号点断掉的最小代 ...
- AC日记——[SDOI2011]消耗战 洛谷 P2495
[SDOI2011]消耗战 思路: 建虚树走树形dp: 代码: #include <bits/stdc++.h> using namespace std; #define INF 1e17 ...
- [BZOJ2286][SDOI2011]消耗战(虚树DP)
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4998 Solved: 1867[Submit][Statu ...
- BZOJ2286 [Sdoi2011]消耗战 【虚树 + 树形Dp】
2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 4261 Solved: 1552 [Submit][Sta ...
- 【BZOJ2286】[Sdoi2011]消耗战 虚树
[BZOJ2286][Sdoi2011]消耗战 Description 在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的 ...
随机推荐
- 详解调试Apache的mod_rewrite模块
大家都知道Apache里面的Rewrite规则是一件很蛋疼的事情,有时候只是想做一个伪静态而已,不想去研究那些复杂的规则,可官方给的规则又常常出错,出了问题我们就要调试一下,看看提交的参数被映射到了哪 ...
- GNS3连接虚拟机
打开GNS3,拖一台路由器和主机 右击主机,选择配置 添加虚拟机的网卡为主机网卡,(选择VM1或VM8) 路由器选择虚拟机网卡连接 打开虚拟机在导航栏找到“虚拟机”-->“设 ...
- init方法返回值自动改写问题
[init方法返回值自动改写问题] 在ARC开启的情况下,以init开头的实例方法的返回值会被默认无视,返回类型会被编译器改写为类指针类型. 如一人类叫UIButton类,如果一个方法叫 (UILab ...
- leetcode:Median of Two Sorted Arrays分析和实现
这个问题的大意是提供两个有序的整数数组A与B,A与B并集的中间数.[1,3]与[2]的中间数为2,因为2能将A与B交集均分.而[1,3]与[2,4]的中间数为2.5,取2与3的平均值.故偶数数目的中间 ...
- Map-making Robots: A Review of the Occupancy Grid Map Algorithm
栅格地图算法:http://www.ikaros-project.org/articles/2008/gridmaps/
- 多个if和一个ifelse的区别
一个程序的要求如下,输入一个学生的数学成绩,如果大于等于60,那么就输出good,如果小于60那么输出not good int a scanf_s("%d",&a) if( ...
- xgboost 调参 !
https://jessesw.com/XG-Boost/ http://blog.csdn.net/u010414589/article/details/51153310
- 数据库 MySQL 之 数据操作
数据库 MySQL 之 数据操作 一.MySQL数据类型介绍 MySQL支持多种类型,大致可以分为四类:数值.字符串类型.日期/时间和其他类型. ①二进制类型 bit[(M)] 二进制位(101001 ...
- 使用EasyUI,关于日期格式的文本框按照正常方式获取不到值的问题
这是个小菜在实际工作中遇到的问题,相信很多EasyUI新手很可能也遇到这样的问题,因此小菜觉得有必要拿出来分享一下. 这个问题要从EasyUI的datebox组件说起,小菜用这个组件的时候,发现用$( ...
- App测试从入门到精通之功能测试
App的功能测试指的是针对软件需求以及用户要求针对APP功能进行测试.简单点理解就是保证App功能的正确性,不要系统出现Bug.让用户用户的舒服,用的爽!好了,我们看下关于App的功能测试要点有哪些? ...