2243: [SDOI2011]染色(树链剖分+线段树)
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,表示x和y之间有一条无向边。
下面 行每行描述一个操作:
“C a b c”表示这是一个染色操作,把节点a到节点b路径上所有点(包括a和b)都染成颜色c;
“Q a b”表示这是一个询问操作,询问节点a到节点b(包括a和b)路径上的颜色段数量。
Output
对于每个询问操作,输出一行答案。
Sample Input
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
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]染色(树链剖分+线段树)的更多相关文章
- 2243: [SDOI2011]染色 树链剖分+线段树染色
给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段), 如“112221”由3段组 ...
- bzoj2243[SDOI2011]染色 树链剖分+线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 9012 Solved: 3375[Submit][Status ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- B20J_2243_[SDOI2011]染色_树链剖分+线段树
B20J_2243_[SDOI2011]染色_树链剖分+线段树 一下午净调这题了,争取晚上多做几道. 题意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成 ...
- BZOJ2243 [SDOI2011]染色(树链剖分+线段树合并)
题目链接 BZOJ2243 树链剖分 $+$ 线段树 线段树每个节点维护$lc$, $rc$, $s$ $lc$代表该区间的最左端的颜色,$rc$代表该区间的最右端的颜色 $s$代表该区间的所有连续颜 ...
- BZOJ2243 (树链剖分+线段树)
Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...
- 【bzoj1959】[Ahoi2005]LANE 航线规划 树链剖分+线段树
题目描述 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由千百万星球构成的Samuel星系. 星际空间站的Samuel II巨型计算 ...
- 【BZOJ-2325】道馆之战 树链剖分 + 线段树
2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec Memory Limit: 256 MBSubmit: 1153 Solved: 421[Submit][Statu ...
- POJ3237 (树链剖分+线段树)
Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...
- bzoj4034 (树链剖分+线段树)
Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...
随机推荐
- php第二十四节课
三级联动 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3 ...
- CentOS7.x的DNS服务的基础配置
一.bind服务器安装 bind:开源.稳定.应用广泛的DNS服务.bind的软件包名bind,服务名称named. 查看是否安装bind, 安装bind包: rpm -qa bind yum -y ...
- noip模拟赛 洗衣
分析:好神的一道题啊.对每棵树建个图跑一下floyd可以有40分,想要打出正解就得对树有比较深的认识了. 每次新生成一棵树都是由两棵树i,j拼成的,答案为原来两棵树的答案和+i中每个点到j中每个点的距 ...
- codevs2449 骑士精神
题目描述 Description 在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位.在任何时候一个骑士都能按照骑士的走法(它可以走到和它横坐标相差为1,纵坐标相差为2或者横坐标 ...
- HUD——1286 找新朋友
思路: 裸的欧拉函数 代码: #include<cstdio> #include<cstring> #include<cstdlib> #include<io ...
- Ubuntu 16.04下FireFox安装Flash插件
下载: https://get.adobe.com/flashplayer/ 选择tar.gz包 解压 sudo tar zxvf flash_player_npapi_linux.x86_64.ta ...
- Ubuntu 16.04下截图工具Shutter
Ubuntu下自带截图工具Screenshot,但是有个缺点是不能对截到的图进行标注,快捷键如下: 截图的升级软件Shutter,具有标注的功能 安装: sudo apt-get install sh ...
- Memcached的Web管理工具MemAdmin(待实践)
Memcached的Web管理工具有很多,但是最好用的应该是MemAdmin.基于PHP5开发,所以部署时要注意环境. 介绍:http://www.junopen.com/memadmin/ 下载:h ...
- SecureCRT 8.0公布
百度搜索到的7.3 注冊码生成器还是能够用于8.0的破解. 破解时,选择手动输入(Enter Licence Manually)产生的代码. 添加了一些特性,我最看重的是: 1. 能够在以下命令窗体 ...
- HDU 5358 多校第6场 First One
First One Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) Tota ...