第一个操作比较麻烦,但可以看出它和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]树点涂色的更多相关文章

  1. [SDOI2017][bzoj4817] 树点涂色 [LCT+线段树]

    题面 传送门 思路 $LCT$ 我们发现,这个1操作,好像非常像$LCT$里面的$Access$啊~ 那么我们尝试把$Access$操作魔改成本题中的涂色 我们令$LCT$中的每一个$splay$链代 ...

  2. 【BZOJ4817】树点涂色(LCT,线段树,树链剖分)

    [BZOJ4817]树点涂色(LCT,线段树,树链剖分) 题面 BZOJ Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义 ...

  3. 【BZOJ4817】[Sdoi2017]树点涂色 LCT+线段树

    [BZOJ4817][Sdoi2017]树点涂色 Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路径的权值是:这条路 ...

  4. [BZOJ4817][SDOI2017]树点涂色(LCT+DFS序线段树)

    4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 692  Solved: 408[Submit][Status ...

  5. 【BZOJ4817】【SDOI2017】树点涂色 [LCT][线段树]

    树点涂色 Time Limit: 10 Sec  Memory Limit: 128 MB[Submit][Status][Discuss] Description Bob有一棵n个点的有根树,其中1 ...

  6. [Bzoj4817] [Sdoi2017]树点涂色 (LCT神题)

    4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 629  Solved: 371[Submit][Status ...

  7. [Sdoi2017]树点涂色 [lct 线段树]

    [Sdoi2017]树点涂色 题意:一棵有根树,支持x到根染成新颜色,求x到y颜色数,求x子树里点到根颜色数最大值 考场发现这个信息是可减的,但是没想到lct 特意设计成lct的形式! 如何求颜色数? ...

  8. 「SDOI2017」树点涂色 解题报告

    「SDOI2017」树点涂色 我sb的不行了 其实一开始有一个类似动态dp的想法 每个点维护到lct树上到最浅点的颜色段数,然后维护一个\(mx_{0,1}\)也就是是否用虚儿子的最大颜色 用个set ...

  9. P3703 [SDOI2017]树点涂色

    P3703 [SDOI2017]树点涂色 链接 分析: 首先对于询问,感觉是线段树维护dfs序,每个点记录到根的颜色个数.第二问差分,第三问区间取max. 那么考虑修改,每次将一个点的颜色变成和父节点 ...

随机推荐

  1. socket编程 ------ BSD socket API

    伯克利套接字(Berkeley sockets),也称为BSD Socket.伯克利套接字的应用编程接口(API)是采用C语言的进程间通信的库,经常用在计算机网络间的通信. BSD Socket的应用 ...

  2. Javascript 的addEventListener()及attachEvent()区别分析

    大家都知道事件的用法就是当某个事件(状况)被触发了之后就会去执行某个Function, 尤其是Javascript, 在当红AJAX的催化下, 了解Javascript的Event用法更加重要, 在这 ...

  3. php模板引擎smarty

    一. smarty的特点 速度:相对于其他模板引擎,速度较快 编译型:在下次访问模板时直接访问编译文件,不再进行模板重新编译 缓存技术:可以将用户最终看到的HTML文件缓存成一个静态HTML 插件技术 ...

  4. idea控制台中文乱码问题解决办法

    解决办法: 打开Intellij的安装的bin目录(D:\Program Files\JetBrains\IntelliJ IDEA 14.0\bin ),找到上图的两个文件(根据你的系统是32位或6 ...

  5. bzoj4302 Hdu 5301 Buildings

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4302 [题解] 出自2015多校-学军 题意大概是给出一个n*m的格子有一个格子(x,y)是 ...

  6. bzoj1499: [NOI2005]瑰丽华尔兹&&codevs1748 单调队列优化dp

    这道题 网上题解还是很多很好的 强烈推荐黄学长 码风真的好看 神犇传送门 学习学习 算是道单调队列优化dp的裸题吧 #include<cstdio> #include<cstring ...

  7. 【Atcoder】ARC083 D - Restoring Road Network

    [算法]图论,最短路? [题意]原图为无向连通图,现给定原图的最短路矩阵,求原图最小边权和,n<=300. [题解]要求最小边权和下,原图的所有边一定是所连两端点的最短路. 那么现在将所有最短路 ...

  8. DotNet 学习笔记 MVC模型

    Model Binding Below is a list of model binding attributes: •[BindRequired]: This attribute adds a mo ...

  9. Apache多网站虚拟目录域名

    一台服务器安装了Apache,如何绑定多个域名或网站呢? 最简单高效的方法如下: 1. 先打开Apache的配置文件httpd.conf,在这个文件, 找到这句:“#Include etc/extra ...

  10. bzoj 2152 点剖分

    比较裸的点剖分,访问到每个重心时,记录一个b数组, 代表当前子树中长度为i的边的数量是b[i],因为是3的倍数, 所以边长mod 3保存就行了,然后记录一个sum数组,代表 当前子树中一个子节点为根的 ...