B20J_2243_[SDOI2011]染色_树链剖分+线段树

一下午净调这题了,争取晚上多做几道。

题意:

给定一棵有n个节点的无根树和m个操作,操作有2类:
1、将节点a到节点b路径上所有点都染成颜色c;
2、询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),
如“112221”由3段组成:“11”、“222”和“1”。
 
分析:
线段树结点维护信息:区间内颜色段数t ,区间左端点颜色lco,区间右端点颜色rco,延迟标记lazy。
上传时lco[pos]=lco[lson];rco[pos]=rco[rson];t[pos]=t[lson]+t[rson];如果左区间右端点和右区间左端点颜色相同,区间段数减一;
查询时先加上区间对应的颜色段数(线段树区间合并),然后每次找到上下端点的颜色(直接单点查找),这个跳链的时候就能找到,如果UC=DC,ans--;
注意每次标记下传的时机。
 
代码
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 1000010
#define M 2062145
#define lson pos<<1
#define rson pos<<1|1
int head[N],to[N<<1],nxt[N<<1],co[N],cnt,n,m,scc;
char ch[10];
int dep[N],fa[N],siz[N],son[N],top[N];
int t[M],lzy[M],idx[N],lco[M],rco[M],a[N];
inline void read(int &x)
{
int f=1;x=0;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-1;s=getchar();}
while(s>='0'&&s<='9'){x=(x<<3)+(x<<1)+s-'0';s=getchar();}
x*=f;
}
inline void add(int u,int v)
{
to[++cnt]=v;
nxt[cnt]=head[u];
head[u]=cnt;
}
inline void dfs1(int x,int y)
{
dep[x]=dep[y]+1;
fa[x]=y;
siz[x]=1;
for(int i=head[x];i;i=nxt[i])
{
if(to[i]!=y)
{
dfs1(to[i],x);
siz[x]+=siz[to[i]];
if(siz[to[i]]>siz[son[x]])
{
son[x]=to[i];
}
}
}
}
inline void dfs2(int x,int t)
{
top[x]=t;
idx[x]=++scc;
if(son[x])dfs2(son[x],t);
for(int i=head[x];i;i=nxt[i])
{
if(to[i]!=fa[x]&&to[i]!=son[x])
{
dfs2(to[i],to[i]);
}
}
}
inline void pud(int pos)
{
if(!lzy[pos])return ;
t[lson]=1;lzy[lson]=lco[lson]=rco[lson]=lzy[pos];
t[rson]=1;lzy[rson]=lco[rson]=rco[rson]=lzy[pos];
lzy[pos]=0;
}
inline void up(int l,int r,int pos,int x,int y,int c)
{
if(x<=l&&y>=r)
{
t[pos]=1;
lco[pos]=c;
rco[pos]=c;
lzy[pos]=c;
return ;
}
pud(pos);
int mid=l+r>>1;
if(x<=mid)up(l,mid,lson,x,y,c);
if(y>mid)up(mid+1,r,rson,x,y,c);
t[pos]=t[lson]+t[rson];
if(lco[rson]==rco[lson])t[pos]--;
lco[pos]=lco[lson];
rco[pos]=rco[rson];
}
inline int query(int l,int r,int pos,int x,int y)
{
pud(pos);
int re=0;
int mid=l+r>>1;
if(x<=l&&y>=r)
{
return t[pos];
}
if(y<=mid) return query(l,mid,lson,x,y);
else if(x>mid) return query(mid+1,r,rson,x,y);
else
{
re=query(l,mid,lson,x,y)+query(mid+1,r,rson,x,y);
if(lco[rson]==rco[lson])re--;
return re;
}
}
inline int col(int l,int r,int x,int pos)
{
pud(pos);
if(l==r)return lco[pos];
int mid=l+r>>1;
if(x<=mid)return col(l,mid,x,lson);
else return col(mid+1,r,x,rson);
}
int main()
{
read(n),read(m);
int x,y,z;
register int i;
for(i=1;i<=n;++i)
{
read(a[i]);
}
for(i=1;i<n;++i)
{
read(x),read(y);
add(x,y);
add(y,x);
}
dfs1(1,0);
dfs2(1,1);
for(i=1;i<=n;++i)up(1,n,1,idx[i],idx[i],a[i]);
while(m--)
{
scanf("%s",ch);
if(ch[0]=='C')
{
read(x),read(y),read(z);
while(top[x]!=top[y])
{
if(dep[top[x]]>dep[top[y]])swap(x,y);
up(1,n,1,idx[top[y]],idx[y],z);
y=fa[top[y]];
}
if(dep[x]<dep[y])swap(x,y);
up(1,n,1,idx[y],idx[x],z);
}
else
{
read(x),read(y);
int ans=0;
while(top[x]!=top[y])
{
if(dep[top[x]]>dep[top[y]])swap(x,y);
int DC=col(1,n,idx[top[y]],1),UC=col(1,n,idx[fa[top[y]]],1);
ans+=query(1,n,1,idx[top[y]],idx[y]);
if(UC==DC)ans--;
y=fa[top[y]];
}
if(dep[x]<dep[y])swap(x,y);
ans+=query(1,n,1,idx[y],idx[x]);
if(!ans)ans=1;
printf("%d\n",ans);
}
}
}

B20J_2243_[SDOI2011]染色_树链剖分+线段树的更多相关文章

  1. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  2. bzoj2243[SDOI2011]染色 树链剖分+线段树

    2243: [SDOI2011]染色 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 9012  Solved: 3375[Submit][Status ...

  3. 2243: [SDOI2011]染色 树链剖分+线段树染色

    给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段), 如“112221”由3段组 ...

  4. BZOJ2243 [SDOI2011]染色(树链剖分+线段树合并)

    题目链接 BZOJ2243 树链剖分 $+$ 线段树 线段树每个节点维护$lc$, $rc$, $s$ $lc$代表该区间的最左端的颜色,$rc$代表该区间的最右端的颜色 $s$代表该区间的所有连续颜 ...

  5. B20J_2836_魔法树_树链剖分+线段树

    B20J_2836_魔法树_树链剖分+线段树 题意: 果树共有N个节点,其中节点0是根节点,每个节点u的父亲记为fa[u].初始时,这个果树的每个节点上都没有果子(即0个果子). Add u v d ...

  6. B20J_3231_[SDOI2014]旅行_树链剖分+线段树

    B20J_3231_[SDOI2014]旅行_树链剖分+线段树 题意: S国有N个城市,编号从1到N.城市间用N-1条双向道路连接,城市信仰不同的宗教,为了方便,我们用不同的正整数代表各种宗教. S国 ...

  7. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  8. 【bzoj1959】[Ahoi2005]LANE 航线规划 树链剖分+线段树

    题目描述 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由千百万星球构成的Samuel星系. 星际空间站的Samuel II巨型计算 ...

  9. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

随机推荐

  1. Spring Boot + Jersey发生FileNotFoundException (No such file or directory)

    我在使用Spring Boot + Jersey 项目,解决了上一篇随笔中的FileNotFoundException,然后又报了一个FileNotFoundException,不过报错信息不一样了 ...

  2. C语言关键字register、extern、static、一些总结,及项目中使用的心得

    首先介绍两个概念: 一.变量的生存周期: 变量从建立到撤销的时间段成变量的生存周期.静态变量,从变量产生到整个程序执行结束.当函数使用变量结束后,变量的存储空间依然存在,变量的值也会随着函数的对其的使 ...

  3. Android开发阅读文档资源

    Android Studio:工具:http://developer.android.com/intl/zh-cn/tools/studio/index.html培训教程:http://develop ...

  4. Codeforces Round #479 (Div. 3) A. Wrong Subtraction

    题目网址:http://codeforces.com/contest/977/problem/A 题解:给你一个数n,进行k次变换,从末尾开始-1,512变成511,511变成510,510会把0消掉 ...

  5. UE4中如何使物体始终朝向摄像头?

    要使物体始终正面朝向摄像头需要用到一个关键节点:Find Look at Rotation 其中Start连接需要旋转的物体位置矢量,Target连接摄像头位置矢量 最后设置SetActorRotat ...

  6. flex与js通信、在浏览器中打开新窗口

    一.flex与js通信(通过flex调用js方法) var urlR:URLRequest = new URLRequest("javascript:test('from flex')&qu ...

  7. Maven管理多模块应用

    穿越至目录: 从0开始,构建前后端分离应用 对于概念的一些理解 Maven的作用 管理模块之间的依赖:根据业务需求,系统会划分很多模块,这些模块彼此之间存在着依赖关系.比如系统管理模块依赖着文件上传模 ...

  8. 高性能网络通信框架 HP-Socket

      HP-Socket 详细介绍 HP-Socket 是一套通用的高性能 TCP/UDP/HTTP 通信框架,包含服务端组件.客户端组件和Agent组件,广泛适用于各种不同应用场景的 TCP/UDP/ ...

  9. 命令行备忘录 cli-memo

    前言 有时候想用一个简洁点儿的备忘录,发现没有简洁好用的,于是就想着开发一个,秉着简洁 的思想,所以连界面都没有,只能通过命令行来操作(尽可能的将命令简化).设计的时候 借鉴了git分支的思想,每个备 ...

  10. TensorFlow-谷歌深度学习库 文件I/O Wrapper

    这篇文章主要介绍一下TensorFlow中相关的文件I/O操作,我们主要使tf.gfile来完成. Exists tf.gfile.Exists(filename) 用来判断一个路径是否存在,如果存在 ...