BZOJ 2243(Link Cut Tree解法)

2243: [SDOI2011]染色

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 7078  Solved: 2641
[ Submit][ Status][ Discuss]

Description

给定一棵有n个节点的无根树和m个操作,操作有2类:

1、将节点a到节点b路径上所有点都染成颜色c

2、询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由3段组成:“11”、“222”和“1”。

请你写一个程序依次完成这m个操作。

Input

第一行包含2个整数nm,分别表示节点数和操作数;

第二行包含n个正整数表示n个节点的初始颜色

下面行每行包含两个整数xy,表示xy之间有一条无向边。

下面行每行描述一个操作:

“C a b c”表示这是一个染色操作,把节点a到节点b路径上所有点(包括ab)都染成颜色c

“Q a b”表示这是一个询问操作,询问节点a到节点b(包括ab)路径上的颜色段数量。

Output

对于每个询问操作,输出一行答案。

Sample Input

6 5

2 2 1 2 1 1

1 2

1 3

2 4

2 5

2 6

Q 3 5

C 2 1 1

Q 3 5

C 5 1 2

Q 3 5

Sample Output

3

1

2

HINT

数N<=10^5,操作数M<=10^5,所有的颜色C为整数且在[0, 10^9]之间。


        本来是想找树链剖分的题目做的,但是看到这题对于树链剖分来说实在是太裸了。而且上个星期打CF(CodeForces),Jon Snow那一轮,最后一题树链剖分写得整个人都不好了,我做到了随机数据可以过,但是毕竟人们可以相互hack,有些BT的数据实在是过不了,心里有阴影,表示我还是太弱(很能体会那些被tourist hack的人的体验)。于是又想起了之前,某(wo)人(ya)说几乎所有树链剖分都可以用Link Cut Tree完成,然后硬着头皮用LCT上。然后发现自己对LCT的理解确实比以前清楚太多(毕竟有了上次的教训)。
        好了,言归正传。维护链上颜色块 段数,对是段数,不是颜色数量!如果是颜色数量,其实这题就变得更难了,因为左右不具有直接合并的性质,重复问题不好解决。既然是段数,那么我们就可以比较左右两段的颜色是否相同来加和统计。
        真的那么简单吗?这里有一个易(wo)错(cuo)点(le),那就是在push_up的时候判断左右颜色是否相同,不是比较父亲和左右儿子的颜色,而是比较父亲和左边最右的点的颜色与父亲和右边最左的点的颜色。因为Link Cut Tree由于用了Splay,会改变节点的顺序,而这个改变对于统计颜色段来说是致命的。故左儿子并不是左边最右的,右儿子也不是。这样的话,我们除了对每个节点设置一个col,还要有lcol和rcol。
        你以为注意了这些就能够撸起袖子敲代码吗?那你太嫩了……LCT里面有一个reverse操作,翻转改变左右儿子。如果你直接copy模板,那就恭喜逆WA了。由于有lcol和rcol,所以在reverse的时候,不仅要交换儿子,两个col也要交换,即做到左右两个区间完全交换。
        说完了这些,就真的没别的了,按照普通的那样做,设置lazy标记、统计sum、下传rev标记,对了要用long long!!!……具体见代码:
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<iomanip>
#include<vector>
#include<queue>
#define file(x) freopen(#x".in","r",stdin),freopen(#x".out","w",stdout);
#define MAX_V 100100
using namespace std;

struct Node
{
	int l,r,fa,col,lazy,lcol,rcol;
	long long sum;
	bool rev;
} tree[MAX_V];

bool root[MAX_V],v[MAX_V];
deque<int> stack,list;
vector<int> g[MAX_V];
int n,m;

inline bool get(int x)
{
	return tree[tree[x].fa].r==x;
}

inline void pushup(int x)
{
	if (x)
	{
		tree[x].sum=tree[tree[x].l].sum+tree[tree[x].r].sum+1;
		if (tree[tree[x].l].rcol==tree[x].col) tree[x].sum--;		//与rcol比较
		if (tree[tree[x].r].lcol==tree[x].col) tree[x].sum--;		//与lcol比较
		if (tree[x].l) tree[x].lcol=tree[tree[x].l].lcol; else tree[x].lcol=tree[x].col;
		if (tree[x].r) tree[x].rcol=tree[tree[x].r].rcol; else tree[x].rcol=tree[x].col;
	}
}

inline void reverse(int x)
{
	if (x)
	{
		swap(tree[x].l,tree[x].r);
		swap(tree[x].lcol,tree[x].rcol);		//千万别忘了交换这两个
		tree[x].rev^=1;
	}
}

inline void update(int x,int w)
{
	if (x)
	{
		tree[x].sum=1;
		tree[x].col=tree[x].lazy=tree[x].lcol=tree[x].rcol=w;
	}
}

inline void pushdown(int x)
{
	if ((x)&&(tree[x].rev))
	{
		reverse(tree[x].l);
		reverse(tree[x].r);
		tree[x].rev=0;
	}
	if ((x)&&(tree[x].lazy))
	{
		update(tree[x].l,tree[x].lazy);
		update(tree[x].r,tree[x].lazy);
		tree[x].lazy=0;
	}
}

inline void rotate(int x)
{
	int fa=tree[x].fa,grand=tree[fa].fa;
	if (get(x))
	{
		tree[fa].r=tree[x].l;
		tree[tree[x].l].fa=fa;
		tree[x].l=fa;
	} else
	{
		tree[fa].l=tree[x].r;
		tree[tree[x].r].fa=fa;
		tree[x].r=fa;
	} 
	tree[x].fa=grand;
	tree[fa].fa=x;
	if (root[fa]) root[x]=1,root[fa]=0;
	else if (grand)
		{
			if (tree[grand].l==fa) tree[grand].l=x;
			else if (tree[grand].r==fa) tree[grand].r=x;
		}
	pushup(fa);
}


inline void splay(int x)
{
	int i=x; stack.clear();
	for(;!root[i];i=tree[i].fa)
		stack.push_back(i);
	stack.push_back(i);
	while (!stack.empty())
	{
		pushdown(stack.back());
		stack.pop_back();
	}
	while (!root[x]) 
	{
		if (!root[tree[x].fa])
		{
			if (get(x)==get(tree[x].fa)) rotate(tree[x].fa);
			                        else rotate(x);
		}
		rotate(x);
	}
	pushup(x);
}

inline void access(int x)
{
	int y=0;
	while (x)
	{
		splay(x);
		root[y]=0;
		root[tree[x].r]=1;
		tree[x].r=y;
		pushup(x);
		y=x,x=tree[x].fa;
	}
}

inline void beroot(int x)
{
	access(x);
	splay(x);
	reverse(x);
}

inline void query(int x,int y)
{
	beroot(x);
	access(y);
	splay(y);
	printf("%lld\n",tree[y].sum);
}

inline void modify(int x,int y,int w)
{
	beroot(x);
	access(y);
	splay(y);
	update(y,w);
	pushup(y);
}

inline void bfs()
{
	list.push_back(1);
	v[1]=1;
	while (!list.empty())
	{
		int i=list.front();
		list.pop_front();
		for(int j=0;j<g[i].size();j++)
			if (!v[g[i][j]])
			{
				list.push_back(g[i][j]);
				v[g[i][j]]=1;
				tree[g[i][j]].fa=i;
			}
	}
}

int main()
{
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)
	{
		scanf("%d",&tree[i].col);
		tree[i].sum=1;
		tree[i].lcol=tree[i].rcol=tree[i].col;
	}	
	for(int i=1;i<n;i++)
	{
		int x,y;
		scanf("%d%d",&x,&y);
		g[x].push_back(y);
		g[y].push_back(x);
	}
	bfs();
	memset(root,1,sizeof(root));
	for(int i=1;i<=m;i++)
	{
		char ch; int x,y,z;
		scanf("%s%d%d",&ch,&x,&y);
		if (ch=='C') 
		{
			scanf("%d",&z);
			modify(x,y,z);
		} else query(x,y);
	}
}


### 回答1: bzoj作为一个计算机竞赛的在线评测系统,不仅可以提供大量的题目供程序员练习和学习,还可以帮助程序员提升算法和编程能力。为了更好地利用bzoj进行题目的学习和刷题,制定一个bzoj做题计划是非常有必要的。 首先,我们需要合理安排时间,每天留出一定的时间来做bzoj的题目。可以根据自己的时间安排,每天挑选适量的题目进行解答。可以先从难度较低的题目开始,逐渐提高难度,这样既能巩固基础知识,又能挑战自己的思维能力。 其次,要有一个计划和目标。可以规划一个每周或每月的题目数量目标,以及每个阶段要学习和掌握的算法知识点。可以根据bzoj的题目分类,如动态规划、图论、贪心算法等,结合自己的实际情况,有针对性地选择题目进行学习。 此外,要充分利用bzoj提供的资源。bzoj网站上有很多高质量的题解和优秀的解题代码,可以参考和学习。还有相关的讨论区,可以与其他程序员交流和讨论,同进步。 最后,要坚持并保持思考。做题不是单纯为了刷数量,更重要的是学会思考和总结。遇到难题时,要有耐心,多思考,多尝试不同的解法。即使不能一次性解出来,也要学会思考和分析解题过程,以及可能出现的错误和优化。 总之,bzoj做题计划的关键在于合理安排时间、制定目标、利用资源、坚持思考。通过有计划的刷题,可以提高算法和编程能力,并培养解决问题的思维习惯,在计算机竞赛中取得更好的成绩。 ### 回答2: bzoj做题计划是指在bzoj这个在线测评系统上制定一套学习和刷题的计划,并且将计划记录在excel表格中。该计划主要包括以下几个方面的内容。 首先是学习目标的设定。通过分析自己的水平和知识缺口,可以设定一个合理的目标,比如每天解决一定数量的题目或者提高特定的算法掌握程度。 其次是题目选择的策略。在excel表格中可以记录下自己选择的题目编号、题目类型和难度等信息。可以根据题目的类型和难度来安排每天的刷题计划,确保自己可以逐步提高技巧和解题能力。 然后是学习进度的记录和管理。将每天的完成情况记录在excel表格中,可以清晰地看到自己的学习进度和任务完成情况。可以使用图表等功能来对学习进度进行可视化展示,更好地管理自己的学习计划。 同时,可以在excel表格的备注栏中记录下每道题目的解题思路、关键点和需要复习的知识点等信息。这样可以方便自己回顾和总结,巩固所学的知识。 最后,可以将excel表格与其他相关资料进行整合,比如算法教材、题目解析和学习笔记等。这样可以形成一个完整的学习档案,方便自己进行系统的学习和复习。 总之,bzoj做题计划excel的制定和记录可以帮助我们更加有条理和高效地进行学习和刷题。通过合理安排学习目标和题目选择策略,记录学习进度和思路,并整合其他学习资料,我们可以提高自己的解题能力,并在bzoj上取得更好的成绩。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值