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. 洛谷——P2090 数字对

    P2090 数字对 题目描述 对于一个数字对(a, b),我们可以通过一次操作将其变为新数字对(a+b, b)或(a, a+b). 给定一正整数n,问最少需要多少次操作可将数字对(1, 1)变为一个数 ...

  2. ORACLE 查看当前用户信息(用户,表视图,索引,表空间,同义词,存储过程,约束条件)

    1.用户 查看当前用户的缺省表空间 SQL>select username,default_tablespace from user_users; 查看当前用户的角色 SQL>select ...

  3. CentOS \Linux 6版本系统命令及其使用详解

    概述 常用的linux命令,分为文件管理.磁盘管理.用户管理.软件管理.系统管理等. 文件管理 ls命令 使用权限:所有使用者 使用方式:ls [-alrtAFR] [name...] 说明:显示指定 ...

  4. 【Python实践-9】将字符串转化为浮点型

    利用map和reduce编写一个str2float函数,把字符串'123.456'转换成浮点数123.456. 思路:计算小数位数--->将字符串中的小数点去掉--->字符串转换为整数-- ...

  5. 在线安全清空慢查询日志slowlog

      mysql> show variables like '%slow_query%';+------------------------------------+--------------- ...

  6. [bzoj3209][花神的数论题] (数位dp+费马小定理)

    Description 背景众所周知,花神多年来凭借无边的神力狂虐各大 OJ.OI.CF.TC …… 当然也包括 CH 啦.描述话说花神这天又来讲课了.课后照例有超级难的神题啦…… 我等蒟蒻又遭殃了. ...

  7. [luoguP1130] 红牌(DP)

    传送门 幼儿园DP. ——代码 #include <cstdio> #include <iostream> ; << ); int a[MAXN][MAXN], f ...

  8. JQuery中如何重置(reset)表单(且清空隐藏域)

    由于JQuery中,提交表单是像下面这样的: 所以,想当然的认为,重置表单,当然就是像下面这样子喽: 但是,不幸的是,这样写的话,会有一个让你很郁闷的结果,那就是,表单无法重置! 后来,上网查了一下, ...

  9. [bzoj1022/poj3480]小约翰的游戏John_博弈论

    小约翰的游戏John 题目大意:Nim游戏.区别在于取走最后一颗石子这输. 注释:$1\le cases \le 500$,$1\le n\le 50$. 想法:anti-SG游戏Colon定理. 如 ...

  10. Hackerrank manasa-and-combinatorics(数学推导)

    题意:有n个字符A,2n个字符B,问你能用这3n个字母组成多少种字符串,使得组成的字符串所有前缀与后缀的B的数目都大于等于A的数目,对答案mod 99991 分析:类似卡特兰数 ans=总方案数-存在 ...