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

5 6
1 2
2 3
3 4
3 5
2 4 5
3 3
1 4
2 4 5
1 5
2 4 5

Sample Output

3
4
2
2

        --by BZOJ

http://www.lydsy.com/JudgeOnline/problem.php?id=4817



大概自己也没想到会回来写这道题吧。

一直对之前见到的,没做出来的题挺抵触的,

不过这是不对的,

题做不出来,也正常,

谁还能把所有的题全做出来呢,

况且,我见到这个题时还不会LCT,

我当时已经打出了树链剖分做法的全部了,

当然爆了一两个点的栈,SD的省选嘛,

然而自己弱是不应该找理由的,

题做不出来就是弱,

没理由的,

那就不应该逃避了,

那么,如果我没有死,你们都将看到我;



如果用LCT记录颜色个数的话,会非常僵,大概只能解决树链剖分解决的问题,

也就是操作三之外的;

但在观察每一个时期树上的颜色情况,发现与LCT的结构类似;

于是考虑利用LCT的结构维护颜色;

对于一二操作;

一就是从某点ACCESS;

二就是从某点空跑ACCESS而不改变splay的组成;

考虑一对三的影响:

发现在ACCESS连接两个点时,父方的点的原儿子的子树到根的颜色数+1,儿子方的点自己的子树颜色树减一;

维护线段树即可;

代码:

 #include<cstdio>
using namespace std;
int max[],mark[],rank[],dep[],size[],a[];
struct dt{
int fa,ch[];
}data[];
struct ss{
int to,next;
}e[];
int first[],num;
int n,m;
void build(int ,int );
void dfs(int ,int ,int );
void builine(int ,int ,int );
void access(int );
int find_top(int );
void splay(int );
void roll(int );
void find(int ,int );
void add(int ,int ,int ,int ,int ,int );
int ans(int ,int ,int ,int ,int );
void down(int );
inline void in(int &ans)
{
ans=;bool p=false;char ch=getchar();
while((ch>'' || ch<'')&&ch!='-') ch=getchar();
if(ch=='-') p=true,ch=getchar();
while(ch<=''&&ch>='') ans=ans*+ch-'',ch=getchar();
if(p) ans=-ans;
}
int main()
{
int i,j,k,x,y;
in(n),in(m);
for(i=;i<=n-;i++){
in(j),in(k);
build(j,k);build(k,j);
}
num=;dfs(,,);
num=;builine(,n,);
for(i=;i<=m;i++){
in(j);
if(j==){
in(x);
access(x);
}
if(j==){
in(x),in(y);
find(x,y);
}
if(j==){
in(x);
y=ans(,n,,rank[x],rank[x]+size[x]-);
printf("%d\n",y);
}
}
return ;
}
void build(int f,int t){
e[++num].next=first[f];
e[num].to=t;
first[f]=num;
}
void dfs(int now,int d,int fa){
int i;
dep[now]=d;size[now]=;rank[now]=++num;a[num]=now;data[now].fa=fa;
for(i=first[now];i;i=e[i].next)
if(e[i].to!=fa){
dfs(e[i].to,d+,now);
size[now]+=size[e[i].to];
}
}
void builine(int l,int r,int nu){
if(l==r){
max[nu]=dep[a[++num]];
return ;
}
int mid=(l+r)>>;
builine(l,mid,nu<<);
builine(mid+,r,nu<<|);
max[nu]=max[nu<<]>max[nu<<|]?max[nu<<]:max[nu<<|];
}
void access(int x){
int y=,z;
while(x){
splay(x);z=find_top(y);
if(z)add(,n,,rank[z],rank[z]+size[z]-,-);
z=data[x].ch[];
data[x].ch[]=y;
y=x;x=data[x].fa;
z=find_top(z);
if(z)add(,n,,rank[z],rank[z]+size[z]-,);
}
}
int find_top(int x){
splay(x);
while(data[x].ch[])
x=data[x].ch[];
return x;
}
void splay(int x){
if(!x)return ;
int fa,fafa;
fa=data[x].fa;fafa=data[fa].fa;
for(;data[fa].ch[]==x||data[fa].ch[]==x;roll(x),fa=data[x].fa,fafa=data[fa].fa){
if(data[fafa].ch[]==fa||data[fafa].ch[]==fa){
if((data[fafa].ch[]==fa)^(data[fa].ch[]==x))
roll(x);
else
roll(fa);
}
}
}
void roll(int now){
int fa=data[now].fa,fafa=data[fa].fa,wh=data[fa].ch[]==now;
data[fa].ch[wh]=data[now].ch[wh^];data[data[fa].ch[wh]].fa=fa;
data[now].ch[wh^]=fa;data[fa].fa=now;
data[now].fa=fafa;
if (data[fafa].ch[]==fa||data[fafa].ch[]==fa)
data[fafa].ch[data[fafa].ch[]==fa]=now;
}
void find(int x,int y){
int ans=;
x=find_top(x);y=find_top(y);
while(x!=y){
if(dep[x]>dep[y]){
splay(x);x=find_top(data[x].fa);ans++;
}
else{
splay(y);y=find_top(data[y].fa);ans++;
}
}
printf("%d\n",ans);
}
void add(int l,int r,int nu,int L,int R,int x){
if(L<=l&&r<=R){
max[nu]+=x;
mark[nu]+=x;
return ;
}
int mid=(l+r)>>;
down(nu);
if(L<=mid)
add(l,mid,nu<<,L,R,x);
if(R>mid)
add(mid+,r,nu<<|,L,R,x);
max[nu]=max[nu<<]>max[nu<<|]?max[nu<<]:max[nu<<|];
}
int ans(int l,int r,int nu,int L,int R){
if(L<=l&&r<=R)
return max[nu];
int mid=(l+r)>>,lm=,rm=;
down(nu);
if(L<=mid)
lm=ans(l,mid,nu<<,L,R);
if(R>mid)
rm=ans(mid+,r,nu<<|,L,R);
if(lm>rm)return lm;
return rm;
}
void down(int nu){
if(!mark[nu])return;
max[nu<<]+=mark[nu];max[nu<<|]+=mark[nu];
mark[nu<<]+=mark[nu];mark[nu<<|]+=mark[nu];
mark[nu]=;
}

SDOI2017 树点涂色——LCT the END的更多相关文章

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

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

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

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

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

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

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

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

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

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

  6. P3703 [SDOI2017]树点涂色 LCT维护颜色+线段树维护dfs序+倍增LCA

    \(\color{#0066ff}{ 题目描述 }\) Bob有一棵\(n\)个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点 ...

  7. BZOJ 4817: [Sdoi2017]树点涂色 LCT+Access的性质+DFS序+线段树

    Code: #include<bits/stdc++.h> #define maxn 200003 #define inf -1000000 using namespace std; vo ...

  8. BZOJ.4817.[SDOI2017]树点涂色(LCT DFS序 线段树)

    题目链接 操作\(1.2\)裸树剖,但是操作\(3\)每个点的答案\(val\)很不好维护.. 如果我们把同种颜色的点划分到同一连通块中,那么向根染色的过程就是Access()! 最初所有点间都是虚边 ...

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

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

随机推荐

  1. (二)Python 装饰器

    1. 函数 在 Python 中,使用关键字 def 和一个函数名以及一个可选的参数列表来定义函数.函数使用 return 关键字来返回值.定义和使用一个最简单的函数例子: >>> ...

  2. re 模块 常规方法使用

    前情提要: re模块主要用于正则,用的好了秒杀一切匹配的规则,这里主要是介绍基本用法 一:元字符 1:\w 匹配字符,包含中文,数字或下划线 l ='早乙女露依 123 是我的 321 心目中的 22 ...

  3. python基础知识梳理----6set 集合的应用

    集合内容简介: set 一: 集合简介 集合set集合是python的一个基本数据类型.一般不是很常用set.中的元素是不重复的.无序的.里里面的元素必须是可hash的tuple,bool),str, ...

  4. Webpack学习错误解决笔记

    错误1:在用npm install 安装模块时,时常会出现没有以下类似的错误 解决方法:右键点击node_modules文件夹,选取属性,将文件夹只读选项去除 错误2:在学习到清理/dist文件夹这块 ...

  5. HDU - 4630 离线处理区间点对问题

    题意:给定\(a[1...n]\),多次询问\([L,R]\)中的任意一对数使得\(gcd(a_i,a_j)\)最大 对于gcd,区间内至少存在两个相同的因子才能作为合法的解,存在两个相同因子且最大就 ...

  6. Mac下安装Fiddler抓包工具(别试了,会报错,没办法使用)

    下载: https://www.telerik.com/download/fiddler 离线版本:(链接: https://pan.baidu.com/s/1hr7f8QK 密码: ukg2) 安装 ...

  7. 单独安装VS2012装mono for android

    直接安装个SQL Server2012,然后就可以单独安装VS2012 来用mono for android了.如果不想装,那就装以下这些东西. 广州-PC286() 11:05:28 和 x86  ...

  8. mono for android 第三课--页面布局(转)

    对于C#程序员来说布局不是什么难事,可是对于我这个新手在mono for android 中布局还是有点小纠结的,不会没关系.慢慢学习.好吧我们开始简单的布局.在之前我们拖拽的控件都是自动的去布局,也 ...

  9. 何为Web App,何为Hybird App

    这些概念听起来很火,当下也很流行,真正理解起来却并非易事.如果让我来全面的解释Web App和Hybird App,我觉得还有些困难. 这篇文章只是我深入了解移动领域开发过程中的不断整理和总结,其中涉 ...

  10. java反射机制_读取properties

    代码: import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * 反射特点: ...