4817: [Sdoi2017]树点涂色


Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 629  Solved: 371
[Submit][Status][Discuss]

Description


Bob有一棵n个点的有根树,其中1号点是根节点。Bob在每个点上涂了颜色,并且每个点上的颜色不同。定义一条路
径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色。Bob可能会进行这几种操作:
1 x:
把点x到根节点的路径上所有的点染上一种没有用过的新颜色。
2 x y:
求x到y的路径的权值。
3 x y:
在以x为根的子树中选择一个点,使得这个点到根节点的路径权值最大,求最大权值。
Bob一共会进行m次操作
 

Input


第一行两个数n,m。
接下来n-1行,每行两个数a,b,表示a与b之间有一条边。
接下来m行,表示操作,格式见题目描述
1<=n,m<=100000
 

Output


每当出现2,3操作,输出一行。
如果是2操作,输出一个数表示路径的权值
如果是3操作,输出一个数表示权值的最大值
 

Sample Input



Sample Output



题解:


其实这道题考的十分妙啊,对LCT的虚实边性质进行了充分的利用。

我们读完题发现操作2可以很快想到树剖,操作3可以维护dfs序在线段树上查询子树信息。
 
但是操作1并不是树剖能解决的;
 
再来看看操作1的特点 :每次只会修改一种从未出现的颜色,并且一定是从当前点修改到根节点。
 
然后神奇的操作就来了 :
 
一开始我们按原树建出lct,只是lct中全部都是虚边,一个点到根节点不同 颜色数量就是它往上到根节点经过虚边数量+1。
 
每次操作1相当于就是以把当前点accees上去,这样同样颜色的就以实边连在一起了。是不是很神奇。。
 
然后再来看看怎么统计答案,我们看看lct中什么时候会变虚实边关系,当然就是access了。
 
access时我们有一句话ch[x][1] = y ,直接把x原来的右儿子覆盖了,变成了新的右儿子,
 
那么y的虚实关系从虚转为了实,只用找到y中最浅的一个点,对它子树全部贡献-1即可。
 
然后原来的右儿子从实变为了虚,找到它最浅的一个点,对它子树全部贡献+1即可。
 
然后我们来看看操作2,是对一条路径进行查询,并不能用树剖做。
 
但是我们发现一个性质,因为每次修改的颜色都是新的颜色,并且是从当前点修改到根节点,设lca(u,v) != u && lca(u,v) != v
 
那么u和v一定颜色不同,我们只需要在线段树上单询ask(u) + ask(v) -  2 * ask(lca(u,v)) +  1 即可
 
但为什么要加那个1
 

 如图,我们查询左下角颜色为1点和右下角的颜色为2的点;当lca颜色和它们不同时发现lca的颜色减多了,要加回来。
 

当lca和其中一个相同时,我们发现还是减多了,还是得加回来。

一条链的情况自己画图也是同理的。这样对于操作2就用线段树轻松维护了。

操作3????不就是线段树dfs序查询子树吗。

这样我们就神奇的利用了LCT的性质把一道看似树剖的题做成了LCT神题。。

AC代码:

过了样例直接交,一遍交过的酸爽


# include <iostream>
# include <cstdio>
# include <cstring>
using namespace std;
const int N = 2e5 + ;
int st[N],ed[N],fa[N],ch[N][],s[N << ],la[N << ],n,m,head[N],dt;
struct Edge{
int to,nex;
}edge[N << ];
void AddEdge(int u,int v)
{
edge[++dt] = (Edge){v,head[u]};
head[u] = dt;
}
bool isroot(int x){return ch[fa[x]][] != x && ch[fa[x]][] != x;}
void rotate(int x,int d)
{
int pre = fa[x],g = fa[pre],nex = ch[x][d];
ch[pre][d ^ ] = nex;
if(nex)fa[nex] = pre;
fa[x] = g;
if(!isroot(pre))ch[g][ch[g][] == pre] = x;
ch[x][d] = pre;
fa[pre] = x;
}
void splay(int x)
{
int pre,g;
while(!isroot(x))
{
pre = fa[x],g = fa[pre];
if(!isroot(pre) && !((ch[pre][] == x) ^ (ch[g][] == pre)))rotate(pre,ch[pre][] == x);
rotate(x,ch[pre][] == x);
}
}
int find(int x){while(ch[x][])x = ch[x][];return x;}
void push(int x){s[x] = max(s[x << ],s[x << | ]);}
void down(int x)
{
s[x << ] += la[x];s[x << | ] += la[x];
la[x << | ] += la[x];la[x << ] += la[x];
la[x] = ;
}
void updata(int L,int R,int l,int r,int rt,int d)
{
if(L <= l && r <= R){s[rt] += d;la[rt] += d;return;}
down(rt);int mid = l + r >> ;
if(L <= mid)updata(L,R,l,mid,rt << ,d);
if(R > mid)updata(L,R,mid + ,r,rt << | ,d);
push(rt);
}
int Query(int L,int R,int l,int r,int rt)
{
if(L <= l && r <= R)return s[rt];
down(rt);int mid = l + r >> ;
if(L > mid)return Query(L,R,mid + ,r,rt << | );
if(R <= mid)return Query(L,R,l,mid,rt << );
return max(Query(L,R,l,mid,rt << ),Query(L,R,mid + ,r,rt << | ));
}
int ask(int x){return Query(st[x],st[x],,n,);}
void access(int x)
{
int y = ,t;
while(x)
{
splay(x);
if(t = find(ch[x][]))updata(st[t],ed[t],,n,,);
ch[x][] = y;if(t = find(y))updata(st[t],ed[t],,n,,-);
y = x;x = fa[x];
}
}
int hson[N],sz[N],tot,top[N],dep[N],Fa[N],id[N];
void dfs(int u)
{
sz[u] = ;
for(int i = head[u];i;i = edge[i].nex)
{
if(sz[edge[i].to])continue;
Fa[edge[i].to] = u;
dep[edge[i].to] = dep[u] + ;
dfs(edge[i].to);
sz[u] += sz[edge[i].to];
if(sz[hson[u]] < sz[edge[i].to])hson[u] = edge[i].to;
}
}
void dfs(int u,int tp)
{
top[u] = tp;st[u] = ++tot;id[tot] = u;
if(hson[u])dfs(hson[u],tp);
for(int i = head[u];i;i = edge[i].nex)
if(!st[edge[i].to])dfs(edge[i].to,edge[i].to);
ed[u] = tot;
}
int lca(int x,int y)
{
while(top[x] != top[y])
{
if(dep[top[x]] < dep[top[y]])swap(x,y);
x = Fa[top[x]];
}
return dep[x] < dep[y] ? x : y;
}
int Q1(int u,int v){return ask(u) + ask(v) - * ask(lca(u,v)) + ;}
int Q2(int u){return Query(st[u],ed[u],,n,);}
int main()
{
scanf("%d %d",&n,&m);int tp,x,y;
for(int i = ;i < n;i++)
{
scanf("%d %d",&x,&y);
AddEdge(x,y);AddEdge(y,x);
}
dfs();dfs(,);
for(int i = ;i <= n;i++)updata(st[i],ed[i],,n,,),fa[i] = Fa[i];
while(m--)
{
scanf("%d",&tp);
if(tp == )scanf("%d",&x),access(x);
if(tp == )scanf("%d %d",&x,&y),printf("%d\n",Q1(x,y));
if(tp == )scanf("%d",&x),printf("%d\n",Q2(x));
}
}

[Bzoj4817] [Sdoi2017]树点涂色 (LCT神题)的更多相关文章

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

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

  2. BZOJ4817: [Sdoi2017]树点涂色(LCT)

    Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. ...

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

    题目描述 Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色.Bob可能会进 ...

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

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

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

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

  6. BZOJ 4817 [SDOI2017]树点涂色 (LCT+线段树维护dfs序)

    题目大意:略 涂色方式明显符合$LCT$里$access$操作的性质,相同颜色的节点在一条深度递增的链上 用$LCT$维护一个树上集合就好 因为它维护了树上集合,所以它别的啥都干不了了 发现树是静态的 ...

  7. 【bzoj4817】树点涂色 LCT+线段树+dfs序

    Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. ...

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

    题目 Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色.Bob可能会进行这 ...

  9. 【bzoj4817】[Sdoi2017]树点涂色 LCT+LCA+线段树

    题目描述 给出一棵n个点,以1为根的有根树,每个点初始染有互不相同的颜色.定义一条路径的权值为路径上的颜色种类数.现有m次操作,每次操作为以下三种之一: 1 x: 把点x到根节点的路径上所有的点染上一 ...

随机推荐

  1. 基于jmeter和shell的接口性能自动化

    基于jmeter和shell的接口性能自动化 1. 总体需求 由于性能测试中涉及的查询接口多,版本迭代频繁,版本更新后自动跑一轮查询业务的性能,可以及时发现一些开发修复bug触发的非预期的bug,利用 ...

  2. js join()和split()方法、reverse() 方法、sort()方法

    ############  join()和split()方法  join() 方法用于把数组中的所有元素放入一个字符串. 元素是通过指定的分隔符进行分隔的. 指定分隔符方法join("#&q ...

  3. HTTP隧道代理

    reGeorg的前身是2008年SensePost在BlackHat USA 2008 的 reDuh延伸与扩展.也是目 前安全从业人员使用最多,范围最广,支持多丰富的一款http隧道.从本质上讲,可 ...

  4. js中重载问题

    在js中是没有重载的  但是  Arguments对象(可以实现模拟重载的效果) 利用arguments对象的length属性,可以获取函数接收的参数的个数 例如: function add(){ i ...

  5. DLL动态库多次加载问题

    原因涉及DLL加载和运行机制,主要有两点:1)DLL动态链接库无法独立运行,必须由一个应用程序进程加载到进程空间后才能使用.加载DLL的进程称为宿主进程.被加载的DLL属于宿主进程,不属于宿主进程内某 ...

  6. zabbix auto discovery

    1.configuration>discovery>create discovery rule ip range:192.168.43.2-254 check: http 80 2.con ...

  7. VR技术在数据中心3D机房中的应用 (下)

    VR技术在数据中心3D机房中的应用 (下) 前面给大家简单科普了一下VR的硬件设备以及VR在各个领域的应用,是不是觉得非常高大上?千言万语概括成一句话,VR能给用户带来前所未有的沉浸感和交互方式,让人 ...

  8. Android流行界面结构——Fragment通过ViewPager(带指示器)嵌套Fragment结构的创建方法详解

    原创文章,转载请注明出处http://www.cnblogs.com/baipengzhan/p/6287213.html 当前Android流行界面结构的一种——Fragment通过ViewPage ...

  9. bzoj3336 Uva10572 Black and White

    题目描述: 数据范围:2<=n,m<=8 题解: 很明显需要状压.但是怎么压不知道,压什么不知道. 然后从条件下手. 条件1要求黑色在一起白色在一起,记录轮廓线很容易做到. 条件2要求不能 ...

  10. Css选择器和JQuery基本编程接口

    使用JQuery之前,首先从官网下载库文件 http://jquery.com/ jquery-2.1.4.js和jquery-2.1.4.min.js,前者是完整无压缩版本,用于开发调试:后者是压缩 ...