bzoj 4817: [Sdoi2017]树点涂色
Description
Input
Output
Sample Input
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
4
2
2
HINT
Source
这个题真的是醉得不行。。。
考虑到第一个操作很烦,但是我们可以用LCT的access来解决这一操作。。。
我们把这个点access的时候,把当前点的原来的重儿子所在的子树权值+1,把新接上来的重儿子的子树的权值-1。。。
(这个直接用线段树来实现。。。)
考虑到每次是染上一个未出现的颜色,可以画一下图来思考这样做的正确性:
只有原来的重儿子的子树的val值要改变(+1),其余儿子的val值是不变(只是换了一种别的颜色而已,总数不变)。。。
而新接上的重儿子的子树内因为少了当前点的颜色而需要-1(原来当前点和新重儿子是不同色的。。。)
注意这些修改都是找到深度最小的点的子树来修改。。。
然后对于第二个操作的查询,就是 查询val[u]+val[v]-2*val[lca(u,v)]+1。。。(因为u,v的颜色不相同,分lca的颜色讨论一下)
// MADE BY QT666
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<iostream>
#include<cstring>
using namespace std;
typedef long long ll;
const int N=300050;
int n,m;
struct tree{
int head[N],to[N],cnt,nxt[N],size[N],son[N],fa[N],top[N],dfn[N],ed[N],tt,deep[N];
void lnk(int x,int y){
to[++cnt]=y,nxt[cnt]=head[x],head[x]=cnt;
to[++cnt]=x,nxt[cnt]=head[y],head[y]=cnt;
}
void dfs1(int x,int f){
size[x]=1;deep[x]=deep[f]+1;
for(int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=f){
fa[y]=x;dfs1(y,x);
size[x]+=size[y];
if(size[y]>size[son[x]]) son[x]=y;
}
}
}
void dfs2(int x,int ff){
top[x]=ff;dfn[x]=++tt;
if(son[x]) dfs2(son[x],ff);
for(int i=head[x];i;i=nxt[i]){
int y=to[i];
if(y!=son[x]&&y!=fa[x]) dfs2(y,y);
}
ed[x]=tt;
}
int lca(int x,int y){
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) swap(x,y);
x=fa[top[x]];
}
if(deep[x]<deep[y]) swap(x,y);
return y;
}
}Tree;
struct segment_tree{
int rt,Max[N],lazy[N],sz,ls[N],rs[N];
void insert(int &x,int l,int r,int v,int d){
if(!x) x=++sz;
if(l==r){Max[x]=d;return;}
int mid=(l+r)>>1;
if(v<=mid) insert(ls[x],l,mid,v,d);
else insert(rs[x],mid+1,r,v,d);
Max[x]=max(Max[ls[x]],Max[rs[x]]);
}
void update(int x,int l,int r,int xl,int xr,int tag){
if(xl<=l&&r<=xr){
Max[x]+=tag;lazy[x]+=tag;return;
}
int mid=(l+r)>>1;
if(xr<=mid) update(ls[x],l,mid,xl,xr,tag);
else if(xl>mid) update(rs[x],mid+1,r,xl,xr,tag);
else update(ls[x],l,mid,xl,mid,tag),update(rs[x],mid+1,r,mid+1,xr,tag);
Max[x]=max(Max[ls[x]],Max[rs[x]])+lazy[x];
}
int query(int x,int l,int r,int xl,int xr,int la){
if(xl<=l&&r<=xr) return Max[x]+la;
int mid=(l+r)>>1;la+=lazy[x];
if(xr<=mid) return query(ls[x],l,mid,xl,xr,la);
else if(xl>mid) return query(rs[x],mid+1,r,xl,xr,la);
else return max(query(ls[x],l,mid,xl,mid,la),query(rs[x],mid+1,r,mid+1,xr,la));
}
}seg;
struct link_cut_tree{
int c[N][2],fa[N];
bool isroot(int x){
return c[fa[x]][0]!=x && c[fa[x]][1]!=x;
}
void rotate(int x){
int y=fa[x],z=fa[y],l,r;
if(c[y][0]==x)l=0;else l=1;r=l^1;
if(!isroot(y)){
if(c[z][0]==y)c[z][0]=x;else c[z][1]=x;
}
fa[x]=z;fa[y]=x;fa[c[x][r]]=y;
c[y][l]=c[x][r];c[x][r]=y;
}
void splay(int x){
while(!isroot(x)){
int y=fa[x],z=fa[y];
if(!isroot(y)){
if((c[y][0]==x)^(c[z][0]==y)) rotate(x);
else rotate(y);
}
rotate(x);
}
}
void access(int x){
int t=0;
while(x){
splay(x);
if(c[x][1]){
int y=c[x][1];
while(c[y][0]) y=c[y][0];
seg.update(seg.rt,1,n,Tree.dfn[y],Tree.ed[y],1);
}
if(t){
int y=t;
while(c[y][0]) y=c[y][0];
seg.update(seg.rt,1,n,Tree.dfn[y],Tree.ed[y],-1);
}
c[x][1]=t;t=x;x=fa[x];
}
}
}LCT;
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<n;i++){
int x,y;scanf("%d%d",&x,&y);
Tree.lnk(x,y);
}
Tree.dfs1(1,0);Tree.dfs2(1,1);
for(int i=1;i<=n;i++) seg.insert(seg.rt,1,n,Tree.dfn[i],Tree.deep[i]);
for(int i=1;i<=n;i++) LCT.fa[i]=Tree.fa[i];
for(int i=1;i<=m;i++){
int type;scanf("%d",&type);
if(type==1){
int x;scanf("%d",&x);LCT.access(x);
}
if(type==2){
int x,y;scanf("%d%d",&x,&y);
int Lca=Tree.lca(x,y);
x=Tree.dfn[x],y=Tree.dfn[y],Lca=Tree.dfn[Lca];
printf("%d\n",seg.query(seg.rt,1,n,x,x,0)+seg.query(seg.rt,1,n,y,y,0)-2*seg.query(seg.rt,1,n,Lca,Lca,0)+1);
}
if(type==3){
int x;scanf("%d",&x);
printf("%d\n",seg.query(seg.rt,1,n,Tree.dfn[x],Tree.ed[x],0));
}
}
return 0;
}
然后对于第三个操作就是查询子树最大值。。。
bzoj 4817: [Sdoi2017]树点涂色的更多相关文章
- BZOJ 4817 [SDOI2017]树点涂色 (LCT+线段树维护dfs序)
题目大意:略 涂色方式明显符合$LCT$里$access$操作的性质,相同颜色的节点在一条深度递增的链上 用$LCT$维护一个树上集合就好 因为它维护了树上集合,所以它别的啥都干不了了 发现树是静态的 ...
- 【刷题】BZOJ 4817 [Sdoi2017]树点涂色
Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. ...
- BZOJ 4817: [Sdoi2017]树点涂色(LCT+树剖+线段树)
题目描述 Bob有一棵 nn 个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. Bob ...
- BZOJ 4817: [Sdoi2017]树点涂色 LCT+Access的性质+DFS序+线段树
Code: #include<bits/stdc++.h> #define maxn 200003 #define inf -1000000 using namespace std; vo ...
- BZOJ.4817.[SDOI2017]树点涂色(LCT DFS序 线段树)
题目链接 操作\(1.2\)裸树剖,但是操作\(3\)每个点的答案\(val\)很不好维护.. 如果我们把同种颜色的点划分到同一连通块中,那么向根染色的过程就是Access()! 最初所有点间都是虚边 ...
- bzoj 4817: [Sdoi2017]树点涂色 LCT+树链剖分+线段树
题目: Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. Bob可能会进 ...
- bzoj 4817: [Sdoi2017]树点涂色【树链剖分+LCT】
非常妙的一道题. 首先对于操作一"把点x到根节点的路径上所有的点染上一种没有用过的新颜色",长得是不是有点像LCT中的access操作?进而发现,如果把同一颜色的点连起来作为LCT ...
- BZOJ 4817 [Sdoi2017]树点涂色 ——LCT 线段树
同BZOJ3779. SDOI出原题,还是弱化版的. 吃枣药丸 #include <map> #include <cmath> #include <queue> # ...
- BZOJ 4817: [Sdoi2017]树点涂色(lct+线段树)
传送门 解题思路 跟重组病毒这道题很像.只是有了一个询问\(2\)的操作,然后询问\(2\)的答案其实就是\(val[x]+val[y]-2*val[lca(x,y)]+1\)(画图理解).剩下的操作 ...
随机推荐
- 项目实战6—Mysql实现企业级日志管理、备份与恢复实战
Mysql实现企业级日志管理.备份与恢复实战 环境背景:随着业务的发展,公司业务和规模不断扩大,网站积累了大量的用户信息和数据,对于一家互联网公司来说,用户和业务数据是根基.一旦公司的数据错乱或者丢失 ...
- 实战-CentOS6.8配置nfs服务
如题 #服务端:请自行配置yum源 命令操作:yum install nfs-utils rpcbind #配置文件编辑:vi /etc/exports /data 0.0.0.0 (rw,sync, ...
- 实战-Mysql主从复制
前言: Mysql内建的复制功能是构建大型高性能应用程序的基础.由于目前mysql的高可用性架构MMM和MHA均建立在复制的基础之上,本文就mysql主从复制进行实战描述,希望对读者提供帮助.之前 服 ...
- 16进制到byte转换
我们经常会看到这样的语法 (byte) 0xAD 0xAD实际是个16进制,转换成二进制为:10101101,转换成10进制是:173,它是个正数 10101101只是int的简写,int由4个byt ...
- Linux send和recv
1.send 函数 int send( SOCKET s, const char FAR *buf, int len, int flags ); 不论是客户还是服务器应用程序都用send函数来向TCP ...
- CentOS6.9-zabbix3.2启动失败原因及页面没有mysql选择项
环境内核信息: [root@zabbix- ~]# uname -a Linux lodboyedu- -.el6.x86_64 # SMP Tue Mar :: UTC x86_64 x86_64 ...
- Progressive Web Apps入门
PC和Mobile开发技术演进 PC方向,从客户端到富客户端,到现在广泛使用的Web. 移动方向,目前主要还是原生应用和Mobile Web,PWA相关技术是未来发展方向. PWA的概念 ...
- django同时查询两张表的数据,合并检索对象返回
原始需求: 1.一篇文章内容分N个版块,每篇文章的版块数量不同. 2.有个文章搜索功能,需要同时搜索标题和内容. 实现思路: 1.由于每篇文章的内容版块数量不同,因此将每个文章的标题和内容分开存入2张 ...
- 从开源项目看 Python 单元测试
我觉得以前在我开发程序的时候,除了文档,可能单元测试是另外一个让我希望别人都写,但是自己又一点都不想写的东西.但是,随着开发程序的增多,以及自己对 Bug 的修改的增多,我发现,UT 在很大程度上是对 ...
- UAC
UAC and Security Shield Icon UAC in Wiki User Account Control (UAC) is a technology and security inf ...