题目链接

发现1操作很像lct中的access,然后它每次染的又是一个新颜色,因此同一个颜色就在同一颗splay里了,且一个点到根的权值val[i]也就是到根路径上虚边的个数,然后看access时会对哪些点的val[i]产生影响。

以下的原儿子表示原来与x在同一颗splay中的儿子 (注意不是splay中x的儿子,是原树中x的儿子,即splay中x的后继)。

当x断开与它原儿子所在splay的连接时,原儿子的子树val都要+1,接上的新儿子的splay,新儿子所在子树val要-1。

对于2操作,相当于统计x~y路径上虚边个数+1,val[x]+val[y]-2*val[lca]+1即是。

对于3操作在子树中找个最大的val即可。

val的话可以用线段树维护。

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<vector>
#include<algorithm>
#include<queue>
#include<cmath>
#include<stack>
#include<map>
#define P puts("lala")
#define cp cerr<<"lala"<<endl
#define ln putchar('\n')
#define sp putchar(' ')
#define pb push_back
#define fi first
#define se second
#define mkp make_pair
using namespace std;
typedef pair<int,int> pii;
inline void read(int &re)
{
char ch=getchar();int g=1;
while(ch<'0'||ch>'9') {if(ch=='-')g=-1;ch=getchar();}
re=0;
while(ch<='9'&&ch>='0') re=(re<<1)+(re<<3)+(ch^48),ch=getchar();
re*=g;
}
typedef long long ll;
inline void read(ll &re)
{
char ch=getchar();ll g=1;
while(ch<'0'||ch>'9') {if(ch=='-')g=-1;ch=getchar();}
re=0;
while(ch<='9'&&ch>='0') re=(re<<1)+(re<<3)+ch-48ll,ch=getchar();
re*=g;
} const int N=100050;
int maxv[N<<2],add[N<<2],fa[N],ch[N][2];
int siz[N],dfn[N],tot=0,is[N],f[N][21],n,dep[N]; int head[N],cnt=0;
struct node
{
int to,next;
}e[N<<1];
inline void add_edge(int x,int y)
{
e[++cnt]=(node){y,head[x]};head[x]=cnt;
e[++cnt]=(node){x,head[y]};head[y]=cnt;
} void build(int o,int l,int r)
{
if(l==r) {maxv[o]=dep[is[l]];return ;}
int mid=l+r>>1;
build(o<<1,l,mid); build(o<<1|1,mid+1,r);
maxv[o]=max(maxv[o<<1],maxv[o<<1|1]);
}
void pushdown(int o)
{
if(add[o])
{
add[o<<1]+=add[o]; add[o<<1|1]+=add[o];
maxv[o<<1]+=add[o]; maxv[o<<1|1]+=add[o];
add[o]=0;
}
}
void update(int o,int l,int r,int x,int y,int k) //add k
{
if(x<=l&&r<=y) {add[o]+=k;maxv[o]+=k;return ;}
pushdown(o);
int mid=l+r>>1;
if(x<=mid) update(o<<1,l,mid,x,y,k);
if(y>mid) update(o<<1|1,mid+1,r,x,y,k);
maxv[o]=max(maxv[o<<1],maxv[o<<1|1]);
}
int query(int o,int l,int r,int x,int y)
{
if(x<=l&&r<=y) return maxv[o];
pushdown(o);
int mid=l+r>>1,ans=0;
if(x<=mid) ans=query(o<<1,l,mid,x,y);
if(y>mid) ans=max(ans,query(o<<1|1,mid+1,r,x,y));
return ans;
} inline bool ge(int x) {return ch[fa[x]][1]==x;}
inline bool isroot(int x) {return ch[fa[x]][1]!=x&&ch[fa[x]][0]!=x;}
inline void rotate(int x)
{
int f=fa[x],g=fa[f],wh=ge(x);
if(!isroot(f)) ch[g][ch[g][1]==f]=x;
ch[f][wh]=ch[x][wh^1]; fa[ch[f][wh]]=f;
ch[x][wh^1]=f; fa[f]=x;
fa[x]=g;
}
void splay(int x)
{
for(int f;(f=fa[x])&&!isroot(x);rotate(x))
if(!isroot(f)) rotate(ge(x)==ge(f)?f:x);
} int nex(int x)
{
x=ch[x][1];
while(ch[x][0]) x=ch[x][0];
return x;
} void access(int x)
{
for(int t=0;x;t=x,x=fa[x])
{
splay(x);
if(ch[x][1])
{
int y=nex(x);
update(1,1,n,dfn[y],dfn[y]+siz[y]-1,1);
}
ch[x][1]=t;
if(ch[x][1])
{
int y=nex(x);
update(1,1,n,dfn[y],dfn[y]+siz[y]-1,-1);
}
}
} void dfs(int u,int dad,int d)
{
fa[u]=dad; f[u][0]=dad;
dfn[u]=++tot; is[tot]=u;
siz[u]=1; dep[u]=d;
for(int i=head[u];i;i=e[i].next)
{
int v=e[i].to;
if(v==dad) continue;
dfs(v,u,d+1);
siz[u]+=siz[v];
}
} int lca(int x,int y)
{
if(dep[x]<dep[y]) swap(x,y);
int d=dep[x]-dep[y];
for(int i=19;i>=0;--i) if(d&(1<<i)) x=f[x][i];
if(x==y) return x;
for(int i=19;i>=0;--i) if(f[x][i]!=f[y][i]) x=f[x][i],y=f[y][i];
return f[x][0];
} int ask(int x,int y)
{
int o=lca(x,y);
return query(1,1,n,dfn[x],dfn[x])+query(1,1,n,dfn[y],dfn[y])
-query(1,1,n,dfn[o],dfn[o])*2;
} int main()
{
#ifndef ONLINE_JUDGE
freopen("1.in","r",stdin);freopen("1.out","w",stdout);
#endif
int i,j,opt,T;
read(n);read(T);
for(i=1;i<n;++i)
{
int x,y;read(x);read(y);
add_edge(x,y);
} dfs(1,0,0);
for(j=1;j<=19;++j) for(i=1;i<=n;++i) f[i][j]=f[f[i][j-1]][j-1]; build(1,1,n);
for(int cas=1;cas<=T;++cas)
{
read(opt);
if(opt==1)
{
int x;read(x);
access(x);
}
else if(opt==2)
{
int x,y;read(x);read(y);
printf("%d\n",ask(x,y)+1);
}
else
{
int x;read(x);
printf("%d\n",query(1,1,n,dfn[x],dfn[x]+siz[x]-1)+1);
}
}
return 0;
}
/* */

SDOI2017树点染色的更多相关文章

  1. SDOI2017 树点染色

    \[SDOI2017 树点染色\] 题目描述 Bob 有一棵 $ n $ 个点的有根树,其中 $ 1 $ 号点是根节点.Bob 在每个节点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是,这 ...

  2. loj2001[SDOI2017]树点染色

    题意:给你一棵树,一开始每个点上的颜色互不相同.三种操作:op1:x到根的路径上的点都染上一种新的颜色.op2:设一条路径的权值为val(x,y),求x到y路径的val.op3:询问x的子树中最大的到 ...

  3. 洛谷3703 [SDOI2017] 树点染色 【LCT】【线段树】

    题目分析: 操作一很明显等价于LCT上的access操作,操作二是常识,操作三转化到dfs序上求最大值也是常识.access的时候顺便在线段树中把对应部分-1,把右子树的子树+1即可. 代码: #in ...

  4. codevs 5963 [SDOI2017]树点染色

     [题解]: #include<cstdio> #include<cstring> #include<iostream> using namespace std; ...

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

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

  6. P3703 [SDOI2017]树点涂色

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

  7. codevs 1191 树轴染色 线段树区间定值,求和

    codevs 1191 树轴染色 Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://www.codevs.cn/problem/1191/ Des ...

  8. 【LG3703】[SDOI2017]树点涂色

    [LG3703][SDOI2017]树点涂色 题面 洛谷 题解 更博辣,更博辣!!! 猪年的第一篇博客 一次只能染根到\(x\),且染的颜色未出现过 这句话是我们解题的关键. 设\(x\)到根的颜色数 ...

  9. HDU3974 Assign the task(多叉树转换为线段+线段树区间染色)

    题目大意:有n个人,给你他们的关系(老板和员工),没有直属上司的人就是整个公司的领导者,这意味着n个人形成一棵树(多叉树).当一个人被分配工作时他会让他的下属也做同样的工作(并且立即停止手头正在做的工 ...

随机推荐

  1. Luogu3694 邦邦的大合唱站队 (状压DP)

    状态由\(从前往后排好的长度\)和\(排好的团队\)决定,\(DP\)方程挺有思考价值的. #include <iostream> #include <cstdio> #inc ...

  2. 开源图编辑库 NebulaGraph VEditor 的设计思路分享

    本文首发于 NebulaGraph 公众号 NebulaGraph VEditor 是一个拥有高性能.高可定制的所见即所得图可视化编辑器前端库. NebulaGraph VEditor 底层基于 SV ...

  3. mybatisplus-sql注入器

    sql注入器 使用mybatisplus只需要继承BaseMapper接口即可使用:但是有新的需求需要扩展BaseMapper里面的功能时可使用sql注入器. 扩展BaseMapper里面的功能 点击 ...

  4. qt C2144 语法错误,需要在类型前添加;(分号)

    可能原因:有部分头文件未以";"结尾.

  5. 第九十九篇:JS闭包

    好家伙,总是要来的,去面对那些晦涩难懂的原理,它就在那里,等着我去搞定它 首先我要去补充一些最基本的概念, 1.什么是内存? 新华字典永远的神, 但这个解释显然不够   去看看百度百科: 内存: CP ...

  6. 03_Django-GET请求和POST请求-设计模式及模板层

    03_Django-GET请求和POST请求-设计模式及模板层 视频:https://www.bilibili.com/video/BV1vK4y1o7jH 博客:https://blog.csdn. ...

  7. KingbaseES 开启事务提交跟踪

    KingbaseESV8R6有个参数 track_commit_timestamp,用来开启跟踪事务提交的时间戳. 配置 编辑kingbase.conf,添加配置如下: track_commit_ti ...

  8. 【读书笔记】C#高级编程 第二十二章 安全性

    (一)身份验证和授权 安全性的两个基本支柱是身份验证和授权.身份验证是标识用户的过程,授权在验证了所标识用户是否可以访问特性资源之后进行的. 1.标识和Principal 使用标识可以验证运行应用程序 ...

  9. 发现tab换成空格不起作用,然后解决如下。

    今天发现把 .vimrc 加了set expandtab之后不起作用,这个本来是把代码中的制表符换成空格,免得不同人的设置不同造成代码缩进混乱. 然后搞了半天搞不定,应该是加载了.vimrc之后又加了 ...

  10. jenkins流水线部署springboot应用到k8s集群(k3s+jenkins+gitee+maven+docker)(2)

    前言:上篇已介绍了jenkins在k3s环境部署,本篇继续上篇讲述流水线构建部署流程 1.从gitlab上拉取代码步骤 在jenkins中,新建一个凭证:Manage Jenkins -> Ma ...