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\)(画图理解).剩下的操作 ...
随机推荐
- ViewPager无限轮播与自定义切换动画
一直在寻求一个能用得长久的ViewPager,寻寻觅觅终于发现,ViewPager有这一个就够了. 注:并非完全原创 先看一下效果: 淡入淡出: 旋转: 无限轮播的ViewPager 主要设计思路(以 ...
- 变位词(0029)-swustoj
变位词(0029)水题 变位词如果两个单词的组成字母完全相同,只是字母的排列顺序不一样,则它们就是变位词,两个单词相同也被认为是变位词.如tea 与eat , nic 与cin, ddc与dcd, a ...
- 排列oj
835:排列 总时间限制: 5000ms 内存限制: 65536kB 描述 题目描述: 大家知道,给出正整数n,则1到n这n个数可以构成n!种排列,把这些排列按照从小到大的顺序(字典顺序)列出,如n= ...
- mysqldump指定编码导出数据
mysqldump指定编码导出数据 第一步,导出旧库 mysqldump --default-character-set=latin1 -uroot -pXXX --database db > ...
- 关于mui header在手机上运行丢失问题
并不需要换header, 只需要把引用的例子自带的CSS文件 app.css.里的两个样式:.mui-plus.mui-android header.mui-bar {display: none;}. ...
- PHP运算符优先级 运算符分类
运算符 运算符是可以通过给出的一或多个值(用编程行话来说,表达式)来产生另一个值(因而整个结构成为一个表达式)的东西. 运算符可按照其能接受几个值来分组.一元运算符只能接受一个值,例如 !(逻辑取反运 ...
- c语言中字符串函数的使用
#include<stdio.h> #include<string.h> /* char s1[]="I am a student"; char s2[20 ...
- EEPlat PaaS 整体方案及技术原理
EEPlat PaaS平台提供了基于元数据驱动的以配置为主的高速开发平台,同一时候提供了完整的多租户实现,各租户拥有自己的独立应用和数据库,租户间实现了应用和数据的全然隔离. EEPlat PaaS平 ...
- 以流方式读写文件:文件菜单打开一个文件,文件内容显示在RichTexBox中,执行复制、剪切、粘贴后,通过文件菜单可以保存修改后的文件。
MainWindow.xaml文件 <Window xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation&q ...
- 获取手机是否root信息
private String isRoot(){ String bool = "Root:false"; try{ if ((!new File("/system/bin ...