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. Xmind的使用

    Xmind是用来学习整理思维的工具

  2. C++输入输出重载

    #include <iostream> using namespace std; class Complex2 { public: Complex2(, ) :_x(x), _y(y){ ...

  3. Gym - 101670H Go Northwest!(CTU Open Contest 2017 思维题+map)

    题目: Go Northwest! is a game usually played in the park main hall when occasional rainy weather disco ...

  4. python lambda简易使用

    基本格式 lambda 变量名:函数表达式 ①直接使用 f=lambda x:x**2 f(3) ②设置函数列表 l=[lambda x:x**2, lambda x:x**3, lambda x:x ...

  5. Address already in use: JVM_Bind:8080错误的解决办法

    解决办法:先到命令行查找8080端口号被那个占用,输入后面的命令:netstat -ano 查到 然后到任务管理器把PID为20904的进程给结束掉,就OK了 -------------------- ...

  6. LINUX-挂载一个文件系统

    mount /dev/hda2 /mnt/hda2 挂载一个叫做hda2的盘 - 确定目录 '/ mnt/hda2' 已经存在 umount /dev/hda2 卸载一个叫做hda2的盘 - 先从挂载 ...

  7. 利用ajax全局设置实现拦截器

    var token = localStorage.getItem("token"); $.ajaxSetup({ dataType: "json", cache ...

  8. 洛谷 4246 BZOJ 1018 [SHOI2008]堵塞的交通

    [题解] 原来线段树还可以这么玩.. 我们用线段树维护连通性.对于一个矩形,我们用4个标记维护4个点的联通情况,再用两个标记维护右边两个点与它们右边的与它们在同一行的点的联通情况. 画图表示,就是 另 ...

  9. springcloud(八):熔断器Hystrix

    熔断器 雪崩效应 在微服务架构中通常会有多个服务层调用,基础服务的故障可能会导致级联故障,进而造成整个系统不可用的情况,这种现象被称为服务雪崩效应.服务雪崩效应是一种因“服务提供者”的不可用导致“服务 ...

  10. Spring核心技术(一)——IoC容器和Bean简介

    IoC容器和Bean简介 这章包括了Spring框架对于IoC规则的实现.Ioc也同DI(依赖注入).而对象是通过构造函数,工厂方法,或者一些Set方法来定义对象之间的依赖的.容器在创建这些Bean对 ...