[DP优化方法之虚树]
首先我们看一篇文章 转自xyz:
给出一棵树.
每次询问选择一些点,求一些东西.这些东西的特点是,许多未选择的点可以通过某种方式剔除而不影响最终结果.
于是就有了建虚树这个技巧.....
我们可以用log级别的时间求出点对间的lca....
那么,对于每个询问我们根据原树的信息重新建树,这棵树中要尽量少地包含未选择节点. 这棵树就叫做虚树.
接下来所说的"树"均指虚树,原来那棵树叫做"原树".
构建过程如下:
按照原树的dfs序号(记为dfn)递增顺序遍历选择的节点. 每次遍历节点都把这个节点插到树上.
首先虚树一定要有一个根. 随便扯一个不会成为询问点的点作根.
维护一个栈,它表示在我们已经(用之前的那些点)构建完毕的虚树上,以最后一个插入的点为端点的DFS链.
设最后插入的点为p(就是栈顶的点),当前遍历到的点为x.我们想把x插入到我们已经构建的树上去.
求出lca(p,x),记为lca.有两种情况:
1.p和x分立在lca的两棵子树下.
2.lca是p.
(为什么lca不能是x?
因为如果lca是x,说明dfn(lca)=dfn(x)<dfn(a),而我们是按照dfs序号遍历的,于是dfn(a)<dfn(x),矛盾.)
对于第二种情况,直接在栈中插入节点x即可,不要连接任何边(后面会说为什么).
对于第一种情况,要仔细分析.
我们是按照dfs序号遍历的(因为很重要所以多说几遍......),有dfn(x)>dfn(p)>dfn(lca).
这说明什么呢? 说明一件很重要的事:我们已经把lca所引领的子树中,p所在的子树全部遍历完了!
简略的证明:如果没有遍历完,那么肯定有一个未加入的点h,满足dfn(h)<dfn(x),
我们按照dfs序号递增顺序遍历的话,应该把h加进来了才能考虑x.
这样,我们就直接构建lca引领的,p所在的那个子树. 我们在退栈的时候构建子树.
p所在的子树如果还有其它部分,它一定在之前就构建好了(所有退栈的点都已经被正确地连入树中了),就剩那条链.
如何正确地把p到lca那部分连进去呢?
设栈顶的节点为p,栈顶第二个节点为q.
重复以下操作:
如果dfn(q)>dfn(lca),可以直接连边q->p,然后退一次栈.
如果dfn(q)=dfn(lca),说明q=lca,直接连边lca->p,此时子树已经构建完毕.
如果dfn(q)<dfn(lca),说明lca被p与q夹在中间,此时连边lca->q,退一次栈,再把lca压入栈.此时子树构建完毕.
如果不理解这样操作的缘由可以画画图.....
最后,为了维护dfs链,要把x压入栈. 整个过程就是这样.....
然后就是我自己的理解了 我觉得我的理解虽然不是很严谨但是很容易懂
其实说白了就是如果我找到一个点不在这条链上 然后我们就跳栈顶的点使得栈顶的点和第二栈顶的点夹着lca 当然有可能第二栈顶的点就是lca 每次跳的时候连边
然后弹掉栈顶的点 如果现在栈顶的点不是lca就把lca塞进去 不是now的点就把now塞进去(这个应该是怕同此询问有重复的点吧 我去掉也ac)
然后的话虚树解决的就是总询问点数很少 询问次数很多的题 然后后面的记得清空就好
top=0; S[++top]=1; Plen=0; P[++Plen]=1;
for(LL i=1;i<=K;i++)
{
LL now=H[i]; LL f=lca(S[top],now);
while(dfn[S[top-1]]>dfn[f]){ins(1,S[top-1],S[top],0); top--;}
if(dfn[S[top]]>dfn[f]){ins(1,f,S[top],0); top--;}
if(S[top]!=f) S[++top]=f,P[++Plen]=f;
S[++top]=now,P[++Plen]=now;
}
while(top>1){ins(1,S[top-1],S[top],0); top--;}
h数组是询问的点要按dfn序排一下
[Sdoi2011消耗战 |
这是一道模版题 找到所有点建完虚树后 然后dp 要删去一些边且费用最小 想一想 真正有用的也就只是这些点还有lca的点 所以的话dp一下 要不选下面点一直到根的最小值的和 要不就选lca到根最小值
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<vector>
#include<climits>
#define Maxn 250010
using namespace std;
typedef long long LL;
struct node{LL x,y,next,d;}edge[][Maxn*]; LL len[],first[][Maxn];
void ins(LL k,LL x,LL y,LL d){len[k]++; edge[k][len[k]].x=x; edge[k][len[k]].y=y; edge[k][len[k]].d=d; edge[k][len[k]].next=first[k][x]; first[k][x]=len[k];}
LL dep[Maxn],fa[Maxn][],minx[Maxn]; LL dfn[Maxn],id=; LL N,M;
void Dfs(LL x,LL f)
{
dfn[x]=++id;
for(LL k=first[][x];k!=-;k=edge[][k].next)
{
LL y=edge[][k].y;
if(y!=f){dep[y]=dep[x]+; fa[y][]=x; minx[y]=min(minx[x],edge[][k].d); Dfs(y,x);}
}
}
LL H[Maxn],K; LL top,S[Maxn];
bool Cmp(const LL &x,const LL &y){return dfn[x]<dfn[y];}
LL lca(LL x,LL y)
{
if(dep[x]<dep[y]) swap(x,y);
LL deep=dep[x]-dep[y];
for(LL i=;i>=;i--) if(deep>=(<<i)){deep-=(<<i); x=fa[x][i];}
if(x==y) return x;
for(LL i=;i>=;i--) if(fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
return fa[x][];
}
LL F[Maxn]; bool C[Maxn];
void DP(LL x)
{
F[x]=minx[x]; if(C[x]) return ; LL tmp=;
for(LL k=first[][x];k!=-;k=edge[][k].next)
{
LL y=edge[][k].y;
DP(y); tmp+=F[y];
}
if(tmp<F[x]) F[x]=tmp;
} LL P[Maxn],Plen;
void Solve()
{
for(LL i=;i<=Plen;i++) first[][P[i]]=-; len[]=;
scanf("%lld",&K); for(LL i=;i<=K;i++) scanf("%lld",&H[i]),C[H[i]]=;
sort(H+,H+K+,Cmp);
top=; S[++top]=; Plen=; P[++Plen]=;
for(LL i=;i<=K;i++)
{
LL now=H[i]; LL f=lca(S[top],now);
while(dfn[S[top-]]>dfn[f]){ins(,S[top-],S[top],); top--;}
if(dfn[S[top]]>dfn[f]){ins(,f,S[top],); top--;}
if(S[top]!=f) S[++top]=f,P[++Plen]=f;
if(S[top]!=now) S[++top]=now,P[++Plen]=now;
}
while(top>){ins(,S[top-],S[top],); top--;}
DP(); printf("%lld\n",F[]);
for(LL i=;i<=K;i++) C[H[i]]=;
}
int main()
{
scanf("%lld",&N); len[]=; memset(first[],-,sizeof(first[]));
for(LL i=;i<N;i++){LL x,y,d; scanf("%lld%lld%lld",&x,&y,&d); ins(,x,y,d); ins(,y,x,d);}
dep[]=; for(LL i=;i<=N;i++) minx[i]=LLONG_MAX; Dfs(,);
for(LL j=;j<=;j++)
for(LL i=;i<=N;i++)
fa[i][j]=fa[fa[i][j-]][j-];
scanf("%lld",&M); len[]=; memset(first[],-,sizeof(first[])); memset(C,,sizeof(C));
for(LL i=;i<=M;i++)
Solve();
return ;
}
[Hnoi2014]世界树 |
这一道题就比较劲了
一些点管辖整个树 这些点是给定的 首先我们建一颗虚树 然后因为虚树上有一些点是lca的 也就是空的 我们要把这些点dp一下看看最近去到哪里
然后的话对于虚树上每两个相临的节点 我们二分这两个节点的链 也就是原树上的链 然后找到中间点 切开之后分别属于那两边
但是我们忽略了一个地方 就是有一些点没有被找过 他们为那些询问点下面的 而且下面没有询问点了 那怎么办呢 这些点肯定是跟着上面祖先选什么我就选什么的 这样的话就在祖先那里统计一下扫过了多少个点 剩下没被扫过的就是祖先的
说起来容易打起来难,这道算是经典题 不做不算是会虚树
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<climits>
#define Maxn 300010
using namespace std;
const int inf=1e9;
struct node
{
int x,y,next,d;
}edge[][Maxn*]; int len[],first[][Maxn];
void ins(int k,int x,int y,int d){len[k]++; edge[k][len[k]].x=x; edge[k][len[k]].y=y; edge[k][len[k]].next=first[k][x]; first[k][x]=len[k];}
int N,Q; int deep[Maxn],size[Maxn],fa[][Maxn]; int dfn[Maxn],id=; int unc[Maxn];
bool Cmp(const int &x,const int &y){return dfn[x]<dfn[y];}
void Dfs(int x,int f)
{
size[x]=; dfn[x]=++id;
for(int k=first[][x];k!=-;k=edge[][k].next)
{
int y=edge[][k].y;
if(y!=f)
{
deep[y]=deep[x]+;
fa[][y]=x;
Dfs(y,x);
size[x]+=size[y];
}
}
}
int lca(int x,int y)
{
if(deep[x]<deep[y]) swap(x,y);
int d=(deep[x]-deep[y]);
for(int i=;i>=;i--) if((<<i)<=d) d-=(<<i),x=fa[i][x];
if(x==y) return x;
for(int i=;i>=;i--) if(fa[i][x]!=fa[i][y]) x=fa[i][x],y=fa[i][y];
return fa[][x];
}
int H[Maxn]; int P[Maxn],S[Maxn],top,plen; bool C[Maxn]; int dis(int x,int y){return deep[x]+deep[y]-*deep[lca(x,y)];} pair<int,int> F1[Maxn],F2[Maxn],G[Maxn];
void Dfs1(int x)
{
if(C[x]) F1[x]=F2[x]=make_pair(,x);
for(int k=first[][x];k!=-;k=edge[][k].next)
{
int y=edge[][k].y; Dfs1(y);
if(!C[x])
{
int D=dis(x,y);
if((F1[x].first>F1[y].first+D)||(F1[x].first==F1[y].first+D&&F1[x].second>F1[y].second)) F2[x]=F1[x],F1[x]=make_pair(F1[y].first+D,F1[y].second);
else if((F2[x].first>F1[y].first+D)||(F2[x].first==F1[y].first+D&&F2[x].second>F1[y].second)) F2[x]=make_pair(F1[y].first+D,F1[y].second);
}
}
}
int F[Maxn];
void Dfs2(int x,int f)
{
if(!C[x])
{
G[x].first=G[f].first+dis(x,f); G[x].second=G[f].second;
if(F1[f].second==F1[x].second)
{
if((F2[f].first+dis(f,x)<G[x].first)||(F2[f].first+dis(f,x)==G[x].first&&F2[f].second<G[x].second))
G[x].second=F2[f].second,G[x].first=F2[f].first+dis(f,x);
}
else
if((F1[f].first+dis(f,x)<G[x].first)||(F1[f].first+dis(f,x)==G[x].first&&F1[f].second<G[x].second))
G[x].second=F1[f].second,G[x].first=F1[f].first+dis(f,x);
}
else G[x]=make_pair(,x); if(C[x]) F[x]=x;
else
{
if(G[x].first<F1[x].first||(G[x].first==F1[x].first&&G[x].second<F1[x].second)) F[x]=G[x].second;
if(G[x].first>F1[x].first||(G[x].first==F1[x].first&&G[x].second>F1[x].second)) F[x]=F1[x].second;
}
for(int k=first[][x];k!=-;k=edge[][k].next)
{
int y=edge[][k].y;
Dfs2(y,x);
}
} int Find(int x,int D){for(int i=;i>=;i--) if(D>=(<<i)) D-=(<<i),x=fa[i][x]; return x;} int ans[Maxn];
void DP(int x)
{
ans[F[x]]++; unc[x]--;
for(int k=first[][x];k!=-;k=edge[][k].next)
{
int y=edge[][k].y; int L=Find(y,deep[y]-deep[x]-); int R=fa[][y]; int sizex=size[L]; unc[x]-=sizex; int ret=x;
if(F[x]!=F[y])
{
if(deep[L]<=deep[R])
{
while(deep[L]<=deep[R])
{
int mid=(deep[L]+deep[R])>>; int midx=Find(y,deep[y]-mid);
int disx=dis(F[x],midx); int disy=dis(F[y],midx);
if(disx>disy||(disx==disy&&F[x]>F[y])) R=Find(y,deep[y]-(mid-));
else if(disx<disy||(disx==disy&&F[x]<F[y])) L=Find(y,deep[y]-(mid+)),ret=midx;
}
ans[F[x]]+=size[Find(y,deep[y]-deep[x]-)]-size[Find(y,deep[y]-deep[ret]-)];
ans[F[y]]+=size[Find(y,deep[y]-deep[ret]-)]-size[y];
}
}
else ans[F[x]]+=size[Find(y,deep[y]-deep[x]-)]-size[y];
}
for(int k=first[][x];k!=-;k=edge[][k].next)
{
int y=edge[][k].y;
DP(y);
}
} int B[Maxn];
void Solve()
{
int K; scanf("%d",&K); for(int i=;i<=K;i++){scanf("%d",&H[i]); B[i]=H[i]; C[H[i]]=;}
sort(H+,H+K+,Cmp); top=; S[top]=; plen=; P[]=;
for(int i=;i<=K;i++)
{
int now=H[i]; int f=lca(now,S[top]);
while(dfn[S[top-]]>dfn[f]) ins(,S[top-],S[top],),top--;
if(dfn[S[top]]>dfn[f]) ins(,f,S[top],),top--;
if(S[top]!=f) S[++top]=f,P[++plen]=f;
if(S[top]!=now) S[++top]=now,P[++plen]=now;
}
while(top>) ins(,S[top-],S[top],),top--;
Dfs1();
Dfs2(,);
for(int i=;i<=plen;i++) unc[P[i]]=size[P[i]];
DP();
for(int i=;i<=plen;i++) ans[F[P[i]]]+=unc[P[i]];
for(int i=;i<=K;i++) printf("%d ",ans[B[i]]); printf("\n");
for(int i=;i<=K;i++) ans[H[i]]=;
for(int i=;i<=K;i++) C[H[i]]=;
for(int i=;i<=plen;i++) F1[P[i]].first=F1[P[i]].second=F2[P[i]].first=F2[P[i]].second=G[P[i]].first=G[P[i]].second=F[P[i]]=inf,first[][P[i]]=-;
len[]=; }
int main()
{
scanf("%d",&N); len[]=; memset(first[],-,sizeof(first[]));
for(int i=;i<N;i++){int x,y; scanf("%d%d",&x,&y); ins(,x,y,); ins(,y,x,);}
Dfs(,); for(int i=;i<=;i++) for(int j=;j<=N;j++) fa[i][j]=fa[i-][fa[i-][j]];
memset(first[],-,sizeof(first[])); len[]=;
for(int i=;i<=N;i++) F1[i].first=F1[i].second=F2[i].first=F2[i].second=G[i].first=G[i].second=F[i]=inf;
for(int i=;i<=N;i++) ans[i]=; scanf("%d",&Q);
for(int i=;i<=Q;i++)
Solve();
return ;
}
/*
10
2 1
3 2
4 3
5 4
6 1
7 3
8 3
9 4
10 1
5
2
6 1
5
2 7 3 6 9
1
8
4
8 7 10 3
5
2 9 3 5 8
*/
[DP优化方法之虚树]的更多相关文章
- DP 优化方法大杂烩 & 做题记录 I.
标 * 的是推荐阅读的部分 / 做的题目. 1. 动态 DP(DDP)算法简介 动态动态规划. 以 P4719 为例讲一讲 ddp: 1.1. 树剖解法 如果没有修改操作,那么可以设计出 DP 方案 ...
- DP 优化方法合集
0. 前言 写完这篇文章后发现自己对于 DP 的优化一窍不通,所以补了补 DP 的一些优化,写篇 blog 总结一下. 1. 单调队列/单调栈优化 1.2 算法介绍 这应该算是最基础的 DP 优化方法 ...
- [总结]一些 DP 优化方法
目录 注意本文未完结 写在前面 矩阵快速幂优化 前缀和优化 two-pointer 优化 决策单调性对一类 1D/1D DP 的优化 \(w(i,j)\) 只含 \(i\) 和 \(j\) 的项--单 ...
- 洛谷P3783 [SDOI2017]天才黑客(前后缀优化建图+虚树+最短路)
题面 传送门 题解 去看\(shadowice\)巨巨写得前后缀优化建图吧 话说我似乎连线段树优化建图的做法都不会 //minamoto #include<bits/stdc++.h> # ...
- [DP优化方法之斜率DP]
什么是斜率dp呢 大概就把一些单调的分组问题 从O(N^2)降到O(N) 具体的话我就不多说了 看论文: http://www.cnblogs.com/ka200812/archive/2012/08 ...
- dp常见优化方法
noip范围内的dp优化方法: 加速状态转移 1.前缀和优化 2.单调队列优化 3.线段树或树状数组优化 精简状态 3:精简状态往往是通过对题目本身性质的分析,去省掉一些冗余的状态.相对以上三条套路性 ...
- bzoj 2286(虚树+树形dp) 虚树模板
树链求并又不会写,学了一发虚树,再也不虚啦~ 2286: [Sdoi2011]消耗战 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 5002 Sol ...
- DP 优化小技巧
收录一些比较冷门的 DP 优化方法. 1. 树上依赖性背包 树上依赖性背包形如在树上选出若干个物品做背包问题,满足这些物品连通.由于 01 背包,多重背包和完全背包均可以在 \(\mathcal{O} ...
- bzoj2286: [Sdoi2011]消耗战 虚树
在一场战争中,战场由n个岛屿和n-1个桥梁组成,保证每两个岛屿间有且仅有一条路径可达.现在,我军已经侦查到敌军的总部在编号为1的岛屿,而且他们已经没有足够多的能源维系战斗,我军胜利在望.已知在其他k个 ...
随机推荐
- DevExpress 学习使用之 PrintSystem
这是来自群里边的一段,收集起来,碎片知识是很珍贵的. 傷心孤影(2072201) 16:14:41导出excel加标题用PrintableComponentLink小宝(462561442) 1 ...
- ios学习笔记之2天来总结
学了2天,小结下. ios的基本代码执行流程: 与java的基本异同: 异: 1.基类:java中Object是所有类的父类,而objective-c的根类为NSObject 2.默认访问类型:jav ...
- A2D规则引擎
A2D规则引擎 写了个简单的规则引擎,普通情况够用了: 比如2家公司有各自的利率计算规则,如下: 在C#方面,没有写在C#的业务逻辑代码中,而是移到了外部规则文件中,如(ACompanyRatePol ...
- 消除Switch...Case的过程
http://www.cnblogs.com/happyframework/p/3300170.html 目录 备注需求第一遍代码(重复的代码)第二遍代码(消除重复)备注 备注返回目录 不要重复自己, ...
- jquery选择器之属性过滤选择器
<style type="text/css"> /*高亮显示*/ .highlight{ background-color: gray } </style> ...
- 我的TDD实践---UnitTest单元测试
我的TDD实践---UnitTest单元测试 “我的TDD实践”系列之UnitTest单元测试 写在前面: 我的TDD实践这几篇文章主要是围绕测试驱动开发所展开的,其中涵盖了一小部分测试理论,更多的则 ...
- 如何使用ssh
如何使用ssh自己的笔记本做不了我的运算,只能依靠办公室的工作站,有时很不方便.所以做了一次远程监控.本想用vnc的,发现怎么都连不上,算了.还是SSH好用.工作站和笔记本都是fedora系统,所以默 ...
- 对dump脱壳的一点思考
对dump脱壳的一点思考 偶然翻了一下手机日历,原来今天是夏至啊,时间过的真快.ISCC的比赛已经持续了2个多月了,我也跟着比赛的那些题目学了2个月.......虽然过程很辛苦,但感觉还是很幸运的,能 ...
- php的数组与字符串的转换函数整理
1.将一个字符串转化为数组 str_split()用于将一个字符串转化为数组 语法: str_split(string,length) //string是必须的,是要分割的字符串: //length是 ...
- javascript操作正则表达式对象的方法总结
//正则表达式对象 /* var s = 'good good study day day up '; var r, re; re = new RegExp('study',"g" ...