[sdoi 2017]树点涂色
传送门
Description
Bob 有一棵\(n\)个点的有根树,其中\(1\)号点是根节点。Bob 在每个节点上涂了颜色,并且每个点上的颜色不同。
定义一条路径的权值是,这条路径上的点(包括起点和终点)共有多少种不同的颜色。
Bob 可能会进行这几种操作:
1 x,把点\(x\)到根节点的路径上的所有的点染上一种没有用过的新颜色;2 x y,求 \(x\) 到 \(y\) 的路径的权值;3 x,在以 \(x\) 为根的子树中选择一个点,使得这个点到根节点的路径权值最大,求最大权值。Bob 一共会进行 \(m\) 次操作。
Solution
发现任何时刻,每种颜色都是一条链,考虑直接用lct维护,
1 x就等同于access x发现每次
1 x操作时,修改的是若干个子树的\(ans\),线段树维护\(dfs\)上的答案
access经过轻边是,对线段树进行区间修改即可祖先答案中的颜色,必然出现在孩子的答案中,所以路径的答案可以看成是:
\[ans_x+ans_y-2*ans_{lca(x,y)}+1
\]一个子树的答案直接区间求最值即可
Code
#include<bits/stdc++.h>
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define reg register
inline int read()
{
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*f;
}
const int MN=1e5+5;
int N,M,L[MN],R[MN],dep[MN],fdfn[MN];
class Seg
{
int lazy[MN<<2],t[MN<<2];
void C(int x,int y){t[x]+=y;lazy[x]+=y;}
void down(int x){if(lazy[x])C(x<<1,lazy[x]),C(x<<1|1,lazy[x]),lazy[x]=0;}
void Modi(int x,int l,int r,int a,int b,int ad)
{
if(l==a&&r==b){C(x,ad);return;}
reg int mid=(l+r)>>1;down(x);
if(b<=mid)Modi(x<<1,l,mid,a,b,ad);
else if(a>mid)Modi(x<<1|1,mid+1,r,a,b,ad);
else Modi(x<<1,l,mid,a,mid,ad),Modi(x<<1|1,mid+1,r,mid+1,b,ad);
t[x]=max(t[x<<1],t[x<<1|1]);
}
int Q1(int x,int l,int r,int a,int b)
{
if(l==a&&r==b)return t[x];
reg int mid=(l+r)>>1;down(x);
if(b<=mid)return Q1(x<<1,l,mid,a,b);
else if(a>mid)return Q1(x<<1|1,mid+1,r,a,b);
else return max(Q1(x<<1,l,mid,a,mid),Q1(x<<1|1,mid+1,r,mid+1,b));
}
public:
void Build(int x,int l,int r)
{
if(l==r) return (void)(t[x]=dep[fdfn[l]]);
reg int mid=(l+r)>>1;
Build(x<<1,l,mid);Build(x<<1|1,mid+1,r);
t[x]=max(t[x<<1],t[x<<1|1]);
}
void md(int x,int ad){if(x)Modi(1,1,N,L[x],R[x],ad);}
int q(int x){return Q1(1,1,N,L[x],L[x]);}
int _q(int x){return Q1(1,1,N,L[x],R[x]);}
}T;
class Link_Cut_Tree
{
int fa[MN],c[MN][2];
bool nrt(int x){return c[fa[x]][0]==x||c[fa[x]][1]==x;}
bool get(int x){return c[fa[x]][1]==x;}
void rtt(int x)
{
int y=fa[x],z=fa[y],l=get(x),r=l^1;if(nrt(y))c[z][get(y)]=x;fa[x]=z;
fa[c[x][r]]=y;c[y][l]=c[x][r];fa[y]=x;c[x][r]=y;
}
void Splay(int x)
{
for(;nrt(x);rtt(x))
if(nrt(fa[x])) rtt(get(fa[x])^get(x)?x:fa[x]);
}
int fir(int x){if(!x) return 0;while(c[x][0])x=c[x][0];return x;}
public:
void acs(int x){reg int i;for(i=0;x;x=fa[i=x])Splay(x),T.md(fir(c[x][1]),1),c[x][1]=i,T.md(fir(i),-1);}
void link(int x,int y){fa[x]=y;}
}lct;
class Tree
{
int fa[MN],mx[MN],siz[MN],top[MN],ind;
std::vector<int> A[MN];
void dfs1(int x,int f)
{
reg int i;siz[x]=1;fa[x]=f;lct.link(x,f);dep[x]=dep[f]+1;
for(i=A[x].size()-1;~i;--i)if(A[x][i]^f)
dfs1(A[x][i],x),siz[x]+=siz[A[x][i]],siz[A[x][i]]>siz[mx[x]]?mx[x]=A[x][i]:0;
}
void dfs2(int x,int tp)
{
L[x]=++ind;fdfn[ind]=x;top[x]=tp;if(mx[x])dfs2(mx[x],tp);reg int i;
for(i=A[x].size()-1;~i;--i)if(A[x][i]!=fa[x]&&A[x][i]!=mx[x])dfs2(A[x][i],A[x][i]);
R[x]=ind;
}
public:
void ins(int x,int y){A[x].push_back(y);A[y].push_back(x);}
void build(){dfs1(1,0);dfs2(1,1);}
int lca(int x,int y)
{
while(top[x]^top[y])
dep[top[x]]>dep[top[y]]?x=fa[top[x]]:y=fa[top[y]];
return dep[x]<dep[y]?x:y;
}
}tree;
int main()
{
N=read();M=read();
reg int i,opt,x,y;
for(i=1;i<N;++i) x=read(),tree.ins(x,read());
tree.build();T.Build(1,1,N);
while(M--)
{
opt=read(),x=read();
if(opt==1)lct.acs(x);
if(opt==2)y=read(),printf("%d\n",T.q(x)+T.q(y)+1-2*T.q(tree.lca(x,y)));
if(opt==3)printf("%d\n",T._q(x));
}
return 0;
}
Blog来自PaperCloud,未经允许,请勿转载,TKS!
[sdoi 2017]树点涂色的更多相关文章
- [BZOJ 4817] [SDOI 2017] 树点涂色
Description Bob有一棵 \(n\) 个点的有根树,其中 \(1\) 号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点 ...
- [Sdoi2017]树点涂色 [lct 线段树]
[Sdoi2017]树点涂色 题意:一棵有根树,支持x到根染成新颜色,求x到y颜色数,求x子树里点到根颜色数最大值 考场发现这个信息是可减的,但是没想到lct 特意设计成lct的形式! 如何求颜色数? ...
- 「SDOI2017」树点涂色 解题报告
「SDOI2017」树点涂色 我sb的不行了 其实一开始有一个类似动态dp的想法 每个点维护到lct树上到最浅点的颜色段数,然后维护一个\(mx_{0,1}\)也就是是否用虚儿子的最大颜色 用个set ...
- P3703 [SDOI2017]树点涂色
P3703 [SDOI2017]树点涂色 链接 分析: 首先对于询问,感觉是线段树维护dfs序,每个点记录到根的颜色个数.第二问差分,第三问区间取max. 那么考虑修改,每次将一个点的颜色变成和父节点 ...
- 【LG3703】[SDOI2017]树点涂色
[LG3703][SDOI2017]树点涂色 题面 洛谷 题解 更博辣,更博辣!!! 猪年的第一篇博客 一次只能染根到\(x\),且染的颜色未出现过 这句话是我们解题的关键. 设\(x\)到根的颜色数 ...
- 【BZOJ4817】树点涂色(LCT,线段树,树链剖分)
[BZOJ4817]树点涂色(LCT,线段树,树链剖分) 题面 BZOJ Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义 ...
- [BZOJ4817][SDOI2017]树点涂色(LCT+DFS序线段树)
4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 692 Solved: 408[Submit][Status ...
- 【BZOJ4817】【SDOI2017】树点涂色 [LCT][线段树]
树点涂色 Time Limit: 10 Sec Memory Limit: 128 MB[Submit][Status][Discuss] Description Bob有一棵n个点的有根树,其中1 ...
- 【BZOJ4817】[Sdoi2017]树点涂色 LCT+线段树
[BZOJ4817][Sdoi2017]树点涂色 Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路径的权值是:这条路 ...
随机推荐
- Eclipse集成Git做团队开发
在日常开发工作中,我们通常使用版本控制软件管理团队的源代码,常用的SVN.Git.与SVN相比,Git有分支的概念,可以从主分支创建开发分支,在开发分支测试没有问题之后,再合并到主分支上去,从而避免了 ...
- 在线java堆栈分析工具
1:工具地址 https://gceasy.io/ft-dashboard-web.jsp 2:在线分析结果
- MAC安装配置maven环境变量
1.下载maven包: 下载链接:
- php fsockopen()方法,简化,异步非阻塞调用
介绍在项目中遇到一个问题,就是php是同步的读取下来的,如果一个方法请求的时间长了一点, 那么整个程序走下去将会遇到阻塞,现在我想触发这个方法,但是又不影响我下下面的程序正常的走下去.查了一上午的方法 ...
- Django 常用的 Web 应用程序工具
Django 提供了多种开发 Web 应用程序所需的常用工具,如:缓存.日志.发送邮件.自定义认证等,更多可参考:<https://docs.djangoproject.com/zh-hans/ ...
- Httpd服务入门知识-Httpd服务常见配置案例之基于客户端来源地址实现访问控制
Httpd服务入门知识-Httpd服务常见配置案例之基于客户端来源地址实现访问控制 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Options 1>.OPTIONS指 ...
- SQL基础篇(MICK)
SQL基础教程(Mick) 数据库和SQL C:\PostgreSQL\9.5\bin\psql.exe -U postgres -d shop 数据库的基本概念 数据库(DB):将大量数据保存起来, ...
- urdf 学习记录
1.URDF(Unified Robot Description Format),统一的机器人描述文件格式.主要用来描述机器人的几何形状,在可视化时(如RViz中)显示出机器人的几何形状.与画图软件( ...
- 201671010406-丁家辉-实验十四 团队项目评审&课程学习总结
实验十四 团队项目评审&课程学习总结 项目 内容 这个作业属于哪个课程 [教师博客主页链接] 这个作业的要求在哪里 [作业链接地址] 作业学习目标 (1)掌握软件项目评审会流程(2)反思总结课 ...
- python 获取当前,上级,上上级路径
import os print '***获取当前目录***' print os.getcwd() print os.path.abspath(os.path.dirname(__file__)) pr ...