[BZOJ4817]树点涂色
第一个操作比较麻烦,但可以看出它和lct里的access操作差不多,所以可以利用lct的性质巧妙维护操作1
直接用lct维护树中同颜色的链(因为染色操作是从$x$染到根所以同颜色的点一定形成一条链),权值就存$x$到根的路径权值$v_x$
对于操作1,在access每一次切换$x$的实儿子时,把原实儿子在splay中最左的点(也就是在原树中在$x$以下且同一条链上深度最浅的点)的子树权值$+1$,把新实儿子在splay中最左的点的子树权值$-1$(每一次切换,原实儿子的子树到根的路径上多了一种颜色,新实儿子的子树到根的路径上少了一种颜色)
对于操作2,容易看出答案是$v_x+v_y-2v_{lca_{x,y}}+1$
对于操作3,直接用线段树维护dfs序,就是查询区间最大值
这个lct的性质用得好哇
#include<stdio.h> void swap(int&a,int&b){a^=b^=a^=b;} int max(int a,int b){return a>b?a:b;} int n,a[100010]; struct seg{ int mx[400010],d[400010]; void pushup(int x){mx[x]=max(mx[x<<1],mx[x<<1|1]);} void build(int l,int r,int x){ if(l==r){ mx[x]=a[l]; return; } int mid=(l+r)>>1; build(l,mid,x<<1); build(mid+1,r,x<<1|1); pushup(x); } void add(int x,int v){ d[x]+=v; mx[x]+=v; } void pushdown(int x){ if(d[x]){ add(x<<1,d[x]); add(x<<1|1,d[x]); d[x]=0; } } void modify(int L,int R,int v,int l,int r,int x){ if(L<=l&&r<=R)return add(x,v); pushdown(x); int mid=(l+r)>>1; if(L<=mid)modify(L,R,v,l,mid,x<<1); if(mid<R)modify(L,R,v,mid+1,r,x<<1|1); pushup(x); } int query(int L,int R,int l,int r,int x){ if(L<=l&&r<=R)return mx[x]; pushdown(x); int mid=(l+r)>>1,ans=-1; if(L<=mid)ans=max(ans,query(L,R,l,mid,x<<1)); if(mid<R)ans=max(ans,query(L,R,mid+1,r,x<<1|1)); return ans; } }s; struct tree{ int h[100010],nex[200010],to[200010],in[100010],ou[100010],fa[100010][17],dep[100010],M; void add(int a,int b){ M++; to[M]=b; nex[M]=h[a]; h[a]=M; } void dfs(int x){ in[x]=++M; for(int i=h[x];i;i=nex[i]){ if(to[i]!=fa[x][0]){ fa[to[i]][0]=x; dep[to[i]]=dep[x]+1; dfs(to[i]); } } ou[x]=M; } void pre(){ int i,j,x,y; for(i=1;i<n;i++){ scanf("%d%d",&x,&y); add(x,y); add(y,x); } dep[1]=1; M=0; dfs(1); for(j=1;j<17;j++){ for(i=1;i<=n;i++)fa[i][j]=fa[fa[i][j-1]][j-1]; } for(i=1;i<=n;i++)a[in[i]]=dep[i]; s.build(1,n,1); } void modify(int x,int v){ s.modify(in[x],ou[x],v,1,n,1); } int get(int x){ return s.query(in[x],in[x],1,n,1); } int sub(int x){ return s.query(in[x],ou[x],1,n,1); } int lca(int x,int y){ int i; if(dep[x]<dep[y])swap(x,y); for(i=16;i>=0;i--){ if(dep[fa[x][i]]>=dep[y])x=fa[x][i]; } if(x==y)return x; for(i=16;i>=0;i--){ if(fa[x][i]!=fa[y][i]){ x=fa[x][i]; y=fa[y][i]; } } return fa[x][0]; } }t; struct lct{ int fa[100010],ch[100010][2],l[100010]; void pre(){ int i; for(i=2;i<=n;i++)fa[i]=t.fa[i][0]; for(i=1;i<=n;i++)l[i]=i; } #define ls ch[x][0] #define rs ch[x][1] void pushup(int x){ if(ls) l[x]=l[ls]; else l[x]=x; } void rot(int x){ int y,z,f,b; y=fa[x]; z=fa[y]; f=ch[y][0]==x; b=ch[x][f]; fa[x]=z; fa[y]=x; if(b)fa[b]=y; ch[x][f]=y; ch[y][f^1]=b; if(ch[z][0]==y)ch[z][0]=x; if(ch[z][1]==y)ch[z][1]=x; pushup(y); pushup(x); } bool isrt(int x){return ch[fa[x]][0]!=x&&ch[fa[x]][1]!=x;} void splay(int x){ int y,z; while(!isrt(x)){ y=fa[x]; z=fa[y]; if(!isrt(y))rot((ch[z][0]==y&&ch[y][0]==x)||(ch[z][1]==y&&ch[y][1]==x)?y:x); rot(x); } } void access(int x){ int y=0; while(x){ splay(x); if(rs)t.modify(l[rs],1); rs=y; if(y)t.modify(l[y],-1); pushup(x); y=x; x=fa[x]; } } }l; int main(){ int m,i,x,y; scanf("%d%d",&n,&m); t.pre(); l.pre(); while(m--){ scanf("%d%d",&i,&x); if(i==1)l.access(x); if(i==2){ scanf("%d",&y); printf("%d\n",t.get(x)+t.get(y)-(t.get(t.lca(x,y))<<1)+1); } if(i==3)printf("%d\n",t.sub(x)); } }
[BZOJ4817]树点涂色的更多相关文章
- [SDOI2017][bzoj4817] 树点涂色 [LCT+线段树]
题面 传送门 思路 $LCT$ 我们发现,这个1操作,好像非常像$LCT$里面的$Access$啊~ 那么我们尝试把$Access$操作魔改成本题中的涂色 我们令$LCT$中的每一个$splay$链代 ...
- 【BZOJ4817】树点涂色(LCT,线段树,树链剖分)
[BZOJ4817]树点涂色(LCT,线段树,树链剖分) 题面 BZOJ Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义 ...
- 【BZOJ4817】[Sdoi2017]树点涂色 LCT+线段树
[BZOJ4817][Sdoi2017]树点涂色 Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路径的权值是:这条路 ...
- [BZOJ4817][SDOI2017]树点涂色(LCT+DFS序线段树)
4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 692 Solved: 408[Submit][Status ...
- 【BZOJ4817】【SDOI2017】树点涂色 [LCT][线段树]
树点涂色 Time Limit: 10 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description Bob有一棵n个点的有根树,其中1 ...
- [Bzoj4817] [Sdoi2017]树点涂色 (LCT神题)
4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 629 Solved: 371[Submit][Status ...
- [Sdoi2017]树点涂色 [lct 线段树]
[Sdoi2017]树点涂色 题意:一棵有根树,支持x到根染成新颜色,求x到y颜色数,求x子树里点到根颜色数最大值 考场发现这个信息是可减的,但是没想到lct 特意设计成lct的形式! 如何求颜色数? ...
- 「SDOI2017」树点涂色 解题报告
「SDOI2017」树点涂色 我sb的不行了 其实一开始有一个类似动态dp的想法 每个点维护到lct树上到最浅点的颜色段数,然后维护一个\(mx_{0,1}\)也就是是否用虚儿子的最大颜色 用个set ...
- P3703 [SDOI2017]树点涂色
P3703 [SDOI2017]树点涂色 链接 分析: 首先对于询问,感觉是线段树维护dfs序,每个点记录到根的颜色个数.第二问差分,第三问区间取max. 那么考虑修改,每次将一个点的颜色变成和父节点 ...
随机推荐
- Codeforces Round #506 (Div. 3) 题解
Codeforces Round #506 (Div. 3) 题目总链接:https://codeforces.com/contest/1029 A. Many Equal Substrings 题意 ...
- fastjson对json操作
fastjson对json字符串JSONObject和JSONArray互相转换操作示例 fastjson的方法: Fastjson API入口类是com.alibaba.fastjson.JSON ...
- Java实现十进制数转十六进制数
Now~Let's begin our second question~ 如何利用Java语言将十进制数字转换成十六进制数字呢? 我第一次编码出来的效果是酱紫的~ /** * */ package c ...
- Python代码规范
一:背景 用于规范化ocp python开发,对于使用python开发的程序使用统一的风格,便于代码的维护 二:python风格规范 分号:不要在行尾加分号,也不要用分号将两条命令放在同一行 括号:宁 ...
- PushState+Ajax 完美实现无刷新
转载自:http://lazynight.me/1897.html 折腾一下PJAX,利用HTML5的新API,实现历史记录的完美导入. 不知道你用没用过Github,里边的目录跳转就是用html5的 ...
- expect 实现本地到远程的scp
expect文件demo 令文件名为test.exp #!/usr/bin/expect -f set timeout -1 set pwd " set src_file [lindex $ ...
- 串的模式匹配算法(求子串位置的定位函数Index(S,T,pos))
串的模式匹配的一般方法如算法4.5(在bo4-1.cpp 中)所示:由主串S 的第pos 个字 符起,检验是否存在子串T.首先令i 等于 pos(i 为S 中当前待比较字符的位序),j 等于 1(j ...
- ES6学习笔记(三)—— Set 和 Map
SetES6提供的数据结构,类似于数组,但是成员的值都是唯一的.(提供了一种数组去重的方法) Set 内部判断两个值是否相同使用的是 'Same-value equality',类似于 ===但是 N ...
- jquery教程-Jquery 获取标签个数 size()函数用法
jquery教程-Jquery 获取标签个数 size()函数用法,size() 方法返回被 jQuery 选择器匹配的元素的数量. 语法 $(selector).size() jQuery ...
- c语言指针学习【转】
前言 近期俄罗斯的陨石.四月的血月.五月北京的飞雪以及天朝各种血腥和混乱,给人一种不详的预感.佛祖说的末法时期,五浊恶世 ,十恶之世,人再无心法约束,道德沦丧,和现在正好吻合.尤其是在天朝,空气,水, ...