2243: [SDOI2011]染色

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 8400  Solved: 3150
[Submit][Status][Discuss]

Description

给定一棵有n个节点的无根树和m个操作,操作有2类:

1、将节点a到节点b路径上所有点都染成颜色c;

2、询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由3段组成:“11”、“222”和“1”。

请你写一个程序依次完成这m个操作。

Input

第一行包含2个整数n和m,分别表示节点数和操作数;

第二行包含n个正整数表示n个节点的初始颜色

下面 行每行包含两个整数x和y,表示xy之间有一条无向边。

下面 行每行描述一个操作:

“C a b c”表示这是一个染色操作,把节点a到节点b路径上所有点(包括a和b)都染成颜色c;

“Q a b”表示这是一个询问操作,询问节点a到节点b(包括a和b)路径上的颜色段数量。

Output

对于每个询问操作,输出一行答案。

Sample Input

6 5

2 2 1 2 1 1

1 2

1 3

2 4

2 5

2 6

Q 3 5

C 2 1 1

Q 3 5

C 5 1 2

Q 3 5

Sample Output

3

1

2

HINT

数N<=10^5,操作数M<=10^5,所有的颜色C为整数且在[0, 10^9]之间。

 
/*
线段树维护左区间右端点和右区间左端点颜色
树链剖分时也查询链顶和父亲的颜色
合并时相同则减一
*/
#include<iostream>
#include<cstdio>
#include<cstring> #define N 1000007 using namespace std;
int head[N],siz[N],fa[N],deep[N];
int pos[N],top[N],col[N],sum[N];
int n,m,ans,cnt,num,tot;
struct edge{int u,v,net;}e[N<<];
struct tree{int l,r,sum,lc,rc,flag;}tr[N<<]; inline int read()
{
int x=,f=;char c=getchar();
while(c>''||c<''){if(c=='-')f=-;c=getchar();}
while(c>=''&&c<=''){x=x*+c-'';c=getchar();}
return x*f;
} inline void add(int u,int v)
{
e[++cnt].v=v;e[cnt].net=head[u];head[u]=cnt;
} inline void pushup(int k)
{
if(tr[k<<|].lc==tr[k<<].rc)
tr[k].sum=tr[k<<].sum+tr[k<<|].sum-;
else tr[k].sum=tr[k<<].sum+tr[k<<|].sum;
tr[k].lc=tr[k<<].lc;tr[k].rc=tr[k<<|].rc;
} inline void pushdown(int k)
{
if(tr[k].l==tr[k].r) return;
tr[k<<].sum=tr[k<<|].sum=;
tr[k<<].lc=tr[k<<].rc=tr[k].flag;
tr[k<<|].lc=tr[k<<|].rc=tr[k].flag;
tr[k<<].flag=tr[k<<|].flag=tr[k].flag;
tr[k].flag=;
} void build(int k,int l,int r)
{
tr[k].l=l;tr[k].r=r;
if(l==r)
{
tr[k].sum=;
tr[k].lc=tr[k].rc=sum[++num];
return;
}
int mid=(l+r)>>;
build(k<<,l,mid);build(k<<|,mid+,r);
pushup(k);
} void change(int k,int l,int r,int c)
{
if(tr[k].l==l && tr[k].r==r)
{
tr[k].sum=;
tr[k].flag=tr[k].lc=tr[k].rc=c;
return;
}
if(tr[k].flag) pushdown(k);
int mid=tr[k].l+tr[k].r>>;
if(r<=mid) change(k<<,l,r,c);
else if(l>mid) change(k<<|,l,r,c);
else change(k<<,l,mid,c),change(k<<|,mid+,r,c);
pushup(k);
} int query(int k,int l,int r)
{
if(tr[k].l==l && tr[k].r==r) return tr[k].sum;
if(tr[k].flag) pushdown(k);
pushup(k);
int mid=tr[k].l+tr[k].r>>;
if(r<=mid) return query(k<<,l,r);
else if(l>mid) return query(k<<|,l,r);
else
{
int ld=query(k<<,l,mid);
int rd=query(k<<|,mid+,r);
if(tr[k<<].rc==tr[k<<|].lc) return ld+rd-;
else return ld+rd;
}
} void dfs1(int u)
{
siz[u]=;
for(int i=head[u];i;i=e[i].net)
{
int v=e[i].v;
if(v==fa[u])continue;
fa[v]=u;deep[v]=deep[u]+;
dfs1(v);siz[u]+=siz[v];
}
} void dfs2(int u,int Top)
{
top[u]=Top;int k=;
pos[u]=++tot;sum[pos[u]]=col[u];
for(int i=head[u];i;i=e[i].net)
{
int v=e[i].v;
if(v==fa[u]) continue;
if(siz[v]>siz[k]) k=v;
}if(k!=) dfs2(k,Top);
else return;
for(int i=head[u];i;i=e[i].net)
{
int v=e[i].v;
if(v==fa[u] || v==k)continue;
dfs2(v,v);
}return;
} int querycolor(int k,int pos)
{
if(tr[k].l==tr[k].r && tr[k].l==pos) return tr[k].lc;
if(tr[k].flag) pushdown(k);
pushup(k);
int mid=tr[k].l+tr[k].r>>;
if(pos<=mid) return querycolor(k<<,pos);
else return querycolor(k<<|,pos);
} int solvequery(int x,int y)
{
ans=;
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])swap(x,y);
ans+=query(,pos[top[x]],pos[x]);
if(querycolor(,pos[top[x]])==querycolor(,pos[fa[top[x]]])) ans--;
x=fa[top[x]];
}
ans+=query(,min(pos[x],pos[y]),max(pos[x],pos[y]));
return ans;
} void solvechange(int x,int y,int c)
{
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])swap(x,y);
change(,pos[top[x]],pos[x],c);
x=fa[top[x]];
}
change(,min(pos[x],pos[y]),max(pos[x],pos[y]),c);
} int main()
{
int x,y,z;char ch[];
n=read();m=read();
for(int i=;i<=n;i++) col[i]=read();
for(int i=;i<n;i++)
{
x=read();y=read();
add(x,y);add(y,x);
}
dfs1();dfs2(,);
build(,,n);
for(int i=;i<=m;i++)
{
scanf("%s",ch);
if(ch[]=='Q')
{
x=read();y=read();
printf("%d\n",solvequery(x,y));
}
else
{
x=read();y=read();z=read();
solvechange(x,y,z);
}
}
return ;
}

2243: [SDOI2011]染色(树链剖分+线段树)的更多相关文章

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

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

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

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

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

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

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

    B20J_2243_[SDOI2011]染色_树链剖分+线段树 一下午净调这题了,争取晚上多做几道. 题意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成 ...

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

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

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

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

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

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

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

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

  9. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  10. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

随机推荐

  1. 15Ajax、JSON

    15Ajax.JSON-2018/07/27 1. ThreadLocal 总结:调用该类的get方法,永远返回当前线程放入的数据.线程局部变量. 保证线程安全 (第二阶段day14后半部分视频以及1 ...

  2. <SpringMvc>入门七 拦截器

    什么是拦截器 1.SpringMVC框架中的拦截器用于 对处理器 进行预处理和后处理的技术. 2.可以定义拦截器链,按照顺序执行. 3.拦截器和过滤器功能类似,区别在 拦截器 过滤器 过滤器是Serv ...

  3. Pycharm Anaconda 安装dlib

    由于采用python3.7安装会出现各种问题,两种解决方法. 1)安装Cmake boost等(不推荐,麻烦且不容易成功). 2)安装Anaconda,创建一个python3.6的环境. 这里使用第二 ...

  4. Multisim破解教程

    转载:http://www.121down.com/article/article_52879.html

  5. i2c中应答信号信号总结

    i2c如果用到主从的关系的时候,需要考虑: give_ack();//从器件发送,来表示占用总线,让sda总线保持低电平. get_ack();//主器件判断是否有器件占用总线,sda有器件占用,是低 ...

  6. java之比较两个日期大小----https://blog.csdn.net/dongfangbaiyun/article/details/51225469

    https://blog.csdn.net/dongfangbaiyun/article/details/51225469 java之比较两个日期大小 最近又用到两个日期大小的比较,因此记录在此,方便 ...

  7. noip模拟赛 洗澡

    分析:首先肯定是要用线性筛把素数全部给筛出来的,然后可以维护一个前缀和数组记录1~i个素数的和,对于每一个询问可以从n到1+k枚举它的右端点,然后利用前缀和统计一个长度为K的区间和,看看是不是满足条件 ...

  8. E - Super Jumping! Jumping! Jumping! DP

    Nowadays, a kind of chess game called “Super Jumping! Jumping! Jumping!” is very popular in HDU. May ...

  9. 22、Java并发性和多线程-Java中的读/写锁

    以下内容转自http://ifeve.com/read-write-locks/: 相比Java中的锁(Locks in Java)里Lock实现,读写锁更复杂一些.假设你的程序中涉及到对一些共享资源 ...

  10. 《Linux内核分析》MOOC课程

    http://www.cnblogs.com/wickedpriest/p/4315189.html