【虚树学习笔记([SDOI2011]消耗战)】
题意
想法
首先我们可以很自然的想到怎么在整棵树上进行求解\(DP\)
很简单 每个点有两个选择 要么对其子树的关键点递归求解 要么自己断开
当然断开的\(cost\)为其到根的最短边权
但我们发现每次都进行一次\(O(n)\)的整棵树\(DP\)我们实在是不太敢用
于是这个时候虚树起到了他的作用
给出定义:虚树是一些关键点按照在原树的祖孙关系构建的树
\(当A在原树中是B的祖先,那么在虚树中任然是\)
同时虚树中不仅有关键点,为了保持其原有的一些信息,一些关键点的\(LCA\)仍是必要存在的
芝士:虚树
我们怎么构建这个虚树呢,我们首先对每个点标号\(dfn\),然后对关键点按标号排序
这样保证我们是一条一条链加进来的
我们逐个加入
1.如果栈为空,或者栈中只有一个元素,那么显然应该:
\(stk[++top]=u;\)
2.取\(lca=LCA(u,stk[top])\),如果\(lca=stk[top]\)则说明\(u\)点应该接着\(stk[top]\)点延长当前的树链.做操作:
\(stk[++top]=u;stk[++top]=u;\)
3.如果\(lca≠stk[top]\)则说明\(u与stk[top]\)分属\(lca\)的两颗不同的子树,且包含\(stk[top]\)的这颗子树应该已经构建完成了,我们需要做的是:
将\(lca\)的包含\(stk[top]\)子树的那部分退栈,并将这部分建边形成虚树.如果\(lca\)不在栈(树链)中,那么要把\(lca\)也加入栈中,保证虚树的结构不出现问题,随后将u加入栈中,以表延长树链.
这里建议画图思考
建完虚树在虚树上跑就行了
代码
#include<iostream>
#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
#define ll long long
ll n,k;
struct P{
int to,next,v;
};
int dfn[50005],mn[50005];
ll cnt = 0;
struct T{
int head[25010];
P e[500010];
void add(ll x,ll y,ll v){
e[++cnt].to = y;
e[cnt].next = head[x];
head[x] = cnt;
e[cnt].v = v;
}
int d[25010],fa[25010],son[25010],s[25010];
void dfs(ll now,ll f){
d[now] = d[f] + 1;
fa[now] = f;
ll maxx = 0;
s[now] = 1;
for(int i = head[now];i;i = e[i].next){
if(e[i].to == f)continue;
mn[e[i].to] = std::min(mn[now],e[i].v);
dfs(e[i].to,now);
s[now] += s[e[i].to];
if(s[e[i].to] > maxx)
maxx = s[e[i].to],son[now] = e[i].to;
}
}
int dfncnt,top[250010];
void dfs2(ll now,ll tp){
dfn[now] = ++dfncnt;
top[now] = tp;
if(!son[now])return ;
dfs2(son[now],tp);
for(int i = head[now];i;i = e[i].next){
if(e[i].to == fa[now] || e[i].to == son[now])continue;
dfs2(e[i].to,e[i].to);
}
}
ll lca(ll x,ll y){
while(top[x]!=top[y]){
if(d[top[x]]<d[top[y]]) x^=y^=x^=y;
x=fa[top[x]];
}
return d[x]>d[y]?y:x;
}
}Q;
ll t;
ll top = 0;
int s[25010];
struct FT{
std::vector<int>son[25010];
void add(ll x,ll y){son[x].push_back(y);}
void ins(ll x){
if(top <= 1) {s[++top] = x;return ;}
ll l = Q.lca(x,s[top]);
if(l == s[top]){return ;}
while(top > 1 && dfn[s[top - 1]] >= dfn[l]){
add(s[top - 1],s[top]);
--top;
}
if(l != s[top]) add(l,s[top]),s[top] = l;
s[++top] = x;
}
ll get(ll now){
if(son[now].size() == 0) return mn[now];
ll ans = 0;
for(register int i = 0;i < son[now].size();++i)
ans += get(son[now][i]);
son[now].clear();
return std::min(ans,(ll)mn[now]);
}
}W;
bool cmp(ll x,ll y){return dfn[x] < dfn[y];}
int num[250010];
int main(){
memset(mn,0x3f,sizeof(mn));
scanf("%lld",&n);
for(int i = 1;i <= n - 1;++i){
ll x,y,v;
scanf("%lld%lld%lld",&x,&y,&v);
Q.add(x,y,v);
Q.add(y,x,v);
}
Q.dfs(1,0);
Q.dfs2(1,1);
scanf("%lld",&t);
while(t -- ){
ll k;
scanf("%lld",&k);
for(int i = 1;i <= k;++i)
scanf("%d",&num[i]);
std::sort(num + 1,num + k + 1,cmp);
s[top = 1] = 1;
for(int i = 1;i <= k;++i)
W.ins(num[i]);
while(top > 0)W.add(s[top - 1],s[top]),top -- ;
std::cout<<W.get(1)<<std::endl;
}
}
【虚树学习笔记([SDOI2011]消耗战)】的更多相关文章
- <虚树+树型DP> SDOI2011消耗战
<虚树+树型DP> SDOI2011消耗战 #include <iostream> #include <cstdio> #include <cstring&g ...
- zkw线段树学习笔记
zkw线段树学习笔记 今天模拟赛线段树被卡常了,由于我自带常数 \(buff\),所以学了下zkw线段树. 平常的线段树无论是修改还是查询,都是从根开始递归找到区间的,而zkw线段树直接从叶子结点开始 ...
- 仙人掌&圆方树学习笔记
仙人掌&圆方树学习笔记 1.仙人掌 圆方树用来干啥? --处理仙人掌的问题. 仙人掌是啥? (图片来自于\(BZOJ1023\)) --也就是任意一条边只会出现在一个环里面. 当然,如果你的图 ...
- 线段树学习笔记(基础&进阶)(一) | P3372 【模板】线段树 1 题解
什么是线段树 线段树是一棵二叉树,每个结点存储需维护的信息,一般用于处理区间最值.区间和等问题. 线段树的用处 对编号连续的一些点进行修改或者统计操作,修改和统计的复杂度都是 O(log n). 基础 ...
- Treap-平衡树学习笔记
平衡树-Treap学习笔记 最近刚学了Treap 发现这种数据结构真的是--妙啊妙啊~~ 咳咳.... 所以发一发博客,也是为了加深蒟蒻自己的理解 顺便帮助一下各位小伙伴们 切入正题 Treap的结构 ...
- JSOI2008 Blue Mary开公司 | 李超线段树学习笔记
题目链接:戳我 这相当于是一个李超线段树的模板qwqwq,题解就不多说了. 代码如下: #include<iostream> #include<cstdio> #include ...
- 算法学习——从bzoj2286开始的虚树学习生活
[原创]转载请标明原作者~ http://www.cnblogs.com/Acheing/ 题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2 ...
- Splay伸展树学习笔记
Splay伸展树 有篇Splay入门必看文章 —— CSDN链接 经典引文 空间效率:O(n) 时间效率:O(log n)插入.查找.删除 创造者:Daniel Sleator 和 Robert Ta ...
- CART分类与回归树 学习笔记
CART:Classification and regression tree,分类与回归树.(是二叉树) CART是决策树的一种,主要由特征选择,树的生成和剪枝三部分组成.它主要用来处理分类和回归问 ...
随机推荐
- leetcode 5/300 最长回文子串 py
目录 题目说明 方法一:动态规划--状态转移方程 方法二:优化中心扩展算法 题目说明 要看明白求得是什么,最长回文字串是指例如cababa中ababa是最长的,不是求回文的部分aba 方法一:动态规划 ...
- HTTP请求如何带参
这两天正好作一份API的接口文档,关于HTTP request如何传递参数不是很清楚,这里转载了他人的文档,让我明白了很多.. http://tomfish88.iteye.com/category/ ...
- UltraSoft - Alpha - Scrum Meeting 1
Date: Apr 06th, 2020. 会议内容为讨论功能规格书和技术规格书的撰写. Scrum 情况汇报 进度情况 组员 负责 昨日进度 后两日任务 CookieLau PM.后端 进行Djan ...
- 自定义注解结合切面和spel表达式
在我们的实际开发中可能存在这么一种情况,当方法参数中的某些条件成立的时候,需要执行一些逻辑处理,比如输出日志.而这些代码可能都是差不多的,那么这个时候就可以结合自定义注解加上切面加上spel表达式进行 ...
- 为什么用于开关电源的开关管一般用MOS管而不是三极管
区别: 1.MOS管损耗比三极管小,导通后压降理论上为0. 2.MOS管为电压驱动型,只需要给电压即可,意思是即便串入一个100K的电阻,只要电压够,MOS管还是能够导通. 3.MOS管的温度特性要比 ...
- 多线程--vthread
vthread中包含两个类: vthread.vthread.pool vthread.vthread.thread 其中class pool的原型如下: class pool(builtins.ob ...
- 设计模式(1-3)-动态代理(WeakCache的运用)
阅读本篇文章前,请事先阅读 理解Java的强引用.软引用.弱引用和虚引用. 看看什么是强引用.什么是弱引用及它们的用途,很必要!!! 上一节讲到,获取对应的代理类时,首先会从缓存中去拿,若拿不到才会去 ...
- (一)FastDFS 高可用集群架构学习---简介
1.什么是FastDFS FastDFS 是余庆老师用c语言编写的一筐开源的分布式文件系统,充分考虑了冗余备份,负载均衡,线性扩容等机制,并注重高可用.高性能等指标,使用FastDFS可以很容易搭建一 ...
- docker中镜像的作用
镜像原理镜像 镜像到底是什么?镜像是一种轻量级.可执行的独立软件包,用来打包软件运行环境和基于运行环境开发的软件,它包含运行某个软件所需的所有内容,包括代码.运行时.库.环境变量和配置文件.1.Uni ...
- 『学了就忘』Linux基础命令 — 35、网络中与其他机器通信的命令
目录 1.write命令 2.wall命令 3.mail 命令 使用1:发送邮件 使用2:查看已经接收的邮件 使用3:发送文件内容 1.write命令 (1)write命令的基本信息 命令名称:wri ...