【Luogu】P3979遥远的国度(树链剖分)
不会换根从暑假开始就困扰我了……拖到现在……
会了还是很激动的。
换根操作事实上不需要(也不能)改树剖本来的dfs序……只是在query上动动手脚……
设全树的集合为G,以root为根,u在原根到新根的链上的子树集合为G',则有查询区间=G-G'……
然后查询的时候就查G-G'就行
#include<cstdio>
#include<cstdlib>
#include<cctype>
#include<algorithm>
#include<cstring>
#define left (rt<<1)
#define right (rt<<1|1)
#define mid ((l+r)>>1)
#define lson l,mid,left
#define rson mid+1,r,right
#define maxn 100020
using namespace std;
inline long long read(){
long long num=,f=;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-') f=-;
ch=getchar();
}
while(isdigit(ch)){
num=num*+ch-'';
ch=getchar();
}
return num*f;
} struct Edge{
int next,to;
}edge[maxn*];
int head[maxn],num;
inline void add(int from,int to){
edge[++num]=(Edge){head[from],to};
head[from]=num;
} int tree[maxn*];
int tag[maxn*];
int size[maxn];
int deep[maxn];
int son[maxn];
int father[maxn];
int top[maxn];
int dfn[maxn];
int back[maxn],ID;
int q[maxn];
int s[maxn][];
int n,m; void find(int x,int fa){
deep[x]=deep[fa]+; size[x]=;
for(int i=head[x];i;i=edge[i].next){
int to=edge[i].to;
if(to==fa) continue;
father[to]=x;
s[to][]=x;
find(to,x);
size[x]+=size[to];
if(son[x]==||size[son[x]]<size[to]) son[x]=to;
}
} void unionn(int x,int Top){
dfn[x]=++ID; back[ID]=x;
top[x]=Top;
if(!son[x]) return;
unionn(son[x],Top);
for(int i=head[x];i;i=edge[i].next){
int to=edge[i].to;
if(to==father[x]||to==son[x]) continue;
unionn(to,to);
}
} inline void pushup(int rt){
tree[rt]=min(tree[left],tree[right]);
} void pushdown(int rt){
if(tag[rt]==-) return;
tag[left]=tag[right]=tag[rt];
tree[left]=tree[right]=tag[rt];
tag[rt]=-;
return;
} void build(int l,int r,int rt){
tag[rt]=-;
if(l==r){
tree[rt]=q[back[l]];
return;
}
build(lson);
build(rson);
pushup(rt);
} void memseg(int from,int to,int num,int l,int r,int rt){
if(from<=l&&to>=r){
tree[rt]=tag[rt]=num;
//printf("%d %d l=%d r=%d %d\n",from,to,l,r,num);
return;
}
pushdown(rt);
if(from<=mid) memseg(from,to,num,lson);
if(to>mid) memseg(from,to,num,rson);
pushup(rt);
return;
} int query(int from,int to,int l,int r,int rt){
if(from<=l&&to>=r) return tree[rt];
pushdown(rt);
int ans=0x7fffffff;
if(from<=mid) ans=min(ans,query(from,to,lson));
if(to>mid) ans=min(ans,query(from,to,rson));
return ans;
} int root; void update(int from,int to,int num){
while(top[from]!=top[to]){
if(deep[top[from]]<deep[top[to]]) swap(from,to);
memseg(dfn[top[from]],dfn[from],num,,n,);
from=father[top[from]];
}
if(deep[from]>deep[to]) swap(from,to);
memseg(dfn[from],dfn[to],num,,n,);
return;
} int LCA(int from,int to){
if(deep[from]<deep[to]) swap(from,to);
int f=deep[from]-deep[to];
for(int i=;(<<i)<=f;++i)
if(f&(<<i)) from=s[from][i];
if(from==to) return from;
for(int i=;i>=;--i){
if(s[from][i]==s[to][i]) continue;
from=s[from][i];
to=s[to][i];
}
return s[from][];
} inline void prepare(){
for(int j=;j<=;++j)
for(int i=;i<=n;++i) s[i][j]=s[s[i][j-]][j-];
} int ask(int o){
if(root==) return query(dfn[o],dfn[o]+size[o]-,,n,);
int lca=LCA(root,o);
if(lca!=o) return query(dfn[o],dfn[o]+size[o]-,,n,);
else{
int now=root;
for(int i=;i>=;--i)
if(deep[s[now][i]]>deep[o]) now=s[now][i];
int ans=0x7fffffff;
if(dfn[now]>) ans=min(ans,query(,dfn[now]-,,n,));
if(dfn[now]+size[now]<=n) ans=min(ans,query(dfn[now]+size[now],n,,n,));
return ans;
}
} int main(){
n=read(),m=read();
for(int i=;i<n;++i){
int from=read(),to=read();
add(from,to);
add(to,from);
}
for(int i=;i<=n;++i) q[i]=read();
root=read();
find(,);
unionn(,);
build(,n,);
prepare();
for(int i=;i<=m;++i){
int opt=read();
if(opt==) root=read();
else if(opt==){
int x=read(),y=read(),z=read();
update(x,y,z);
}
else{
int x=read();
printf("%d\n",ask(x));
}
}
return ;
}
/*
10 10
1 2
2 3
2 4
1 5
5 6
5 10
5 7
7 8
7 9
5 1 2 3 6 4 7 8 9 10
1
*/
话说写博客超简略的我简直是业界毒瘤啊……
【Luogu】P3979遥远的国度(树链剖分)的更多相关文章
- 洛谷P3979 遥远的国度 树链剖分+分类讨论
题意:给出一棵树,这棵树每个点有权值,然后有3种操作.操作一:修改树根为rt,操作二:修改u到v路径上点权值为w,操作三:询问以rt为根x子树的最小权值. 解法:如果没有修改树根操作那么这题就是树链剖 ...
- [luogu P3384] [模板]树链剖分
[luogu P3384] [模板]树链剖分 题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点 ...
- BZOJ 3083 遥远的国度 树链剖分
3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 797 Solved: 181[Submit][Status] Descrip ...
- BZOJ 3083 遥远的国度(树链剖分+LCA)
Description 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要z ...
- 【bzoj3083】遥远的国度 树链剖分+线段树
题目描述 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn ...
- BZOJ 3083: 遥远的国度(树链剖分+DFS序)
可以很显而易见的看出,修改就是树链剖分,而询问就是在dfs出的线段树里查询最小值,但由于这道题会修改根节点,所以在查询的时候需判断x是否为root的祖先,如果不是就直接做,是的话应该查询从1-st[y ...
- BZOJ 3083 遥远的国度 树链剖分+线段树
有换根的树链剖分的裸题. 在换根的时候注意讨论. 注意数据范围要开unsigned int或longlong #include<iostream> #include<cstdio&g ...
- 【Luogu P3384】树链剖分模板
树链剖分的基本思想是把一棵树剖分成若干条链,再利用线段树等数据结构维护相关数据,可以非常暴力优雅地解决很多问题. 树链剖分中的几个基本概念: 重儿子:对于当前节点的所有儿子中,子树大小最大的一个儿子就 ...
- BZOJ 3083: 遥远的国度 [树链剖分 DFS序 LCA]
3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 1280 MBSubmit: 3127 Solved: 795[Submit][Status][Discu ...
- luoguP3979 遥远的国度 树链剖分
\(1, 2\)操作没什么好说的 对于\(3\)操作,分三种情况讨论下 \(id = rt\)的情况下,查整棵树的最小值即可 如果\(rt\)在\(1\)号点为根的情况下不在\(id\)的子树中,那么 ...
随机推荐
- 用Jersey为Android客户端开发Restful Web Service
平时在做Android客户端的时候经常要与服务器之间通信,客户端通过服务端提供的接口获取数据,然后再展示在客户端的界面上,作为Android开发者,我们平时更多的是关注客户端的开发,而对服务端开发的关 ...
- MVC的验证码
后台: /// <summary> /// 创建验证码的图片 /// </summary> /// <param name="validateCode" ...
- UVA 1623 Enther the Dragon 神龙喝水 (贪心)
贪心,每次遇到一个满水的湖要下暴雨的时候,就往前找之前最后一次满水之后的第一个没有下雨的且没有被用掉天day1. 因为如果不选这day1,那么之后的湖不一定能选上这一天.如果这一天后面还有没有下雨的天 ...
- js原型,原型链的理解
1.所有引用类型(函数.数组.对象)都拥有_proto_属性(隐式原型) 2.所有函数拥有prototype属性(显式原型)(仅限函数) 3.原型对象:拥有prototype属性的对象,在定义函数时就 ...
- Problem Q: C语言习题 计算该日在本年中是第几天
Problem Q: C语言习题 计算该日在本年中是第几天 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 4572 Solved: 2474[Subm ...
- 1968: C/C++经典程序训练6---歌德巴赫猜想的证明
1968: C/C++经典程序训练6---歌德巴赫猜想的证明 Time Limit: 1 Sec Memory Limit: 64 MBSubmit: 1165 Solved: 499[Submi ...
- Asp.Net Core 进阶(四)—— 过滤器 Filters
一.介绍 Asp.Net Core Filter 使得可以在请求处理管道的特定阶段的前后执行代码,我们可以创建自定义的 filter 用于处理横切关注点. 横切关注点的示例包括错误处理.缓存.配置.授 ...
- 字符编码:BSTR
typedef wchar_t WCHAR; typedef WCHAR OLECHAR; typedef OLECHAR __RPC_FAR *BSTR;;
- NOIP模拟赛 篮球比赛2
篮球比赛2(basketball2.*) 由于Czhou举行了众多noip模拟赛,也导致放学后篮球比赛次数急剧增加.神牛们身体素质突飞猛进,并且球技不断精进.这引起了体育老师彩哥的注意,为了给校篮球队 ...
- springboot下https证书配置
没有证书的小伙伴首先申请一个阿里云免费证书,按照我的步骤来操作 1.购买页面是这样的 按照顺序选择 神奇的一幕出现了 然后就去购买成功,我们会看到证书没有签发,我们需要去申请 填写需要绑定的域名 一般 ...