树链剖分【p2568】[SDOI2011]染色
Description
给定一颗有\(n\)个节点的无根树和\(m\)个操作,操作有\(2\)类:
- 1.将节点\(a\)到节点\(b\)路径上所有点染成颜色\(c\)
- 2.询问节点\(a\)到节点\(b\)路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由三段组成:"11","222"和“1”
请你写一个程序依次完成这\(m\)个操作.
Input
第一行包含两个整数\(n\)和\(m\),分别表示节点数和操作数.
第二行包含\(n\)个正整数表示\(n\)个节点的初始颜色.
下面\(n-1\)行每行包含两个整数\(x\)和\(y\),表示\(x\)和\(y\)之间有一条无向边.
下面\(m\)行每行描述一个操作:
"C a b c",表示这是一个染色操作,把节点\(a\)到节点\(b\)路径上所有点(包括\(a\)和\(b\))都染成颜色\(c\)
“Q a b”表示这是一个询问操作,询问节点\(a\)到节点\(b\)(包括\(a\)和\(b\))路径上的颜色段数量.
Output
对于每个询问操作,输出一行答案。
很明显,这是一个树剖题.
但是维护起来会很麻烦.qwq
首先明确我们的线段树需要维护什么东西.
- 我们需要维护颜色段的个数(题目要求啊,(#`O′))
由于线段树是每次将区间分成一半,所以我们需要考虑拼接。
因此我们需要维护的东西还有
- 当前节点的左子区间中的颜色段
- 当前节点的右子区间中的颜色段.
如果当前节点的左子区间的右子颜色段与当前节点的右子区间的左子颜色段相同,我们就需要拼接.(可能会有些难理解,请细细品读)
然后难点就在于剖分成链之后,我们跳转的时候,链顶与新的一部分的拼接.
这里以链的下端为\(L\),链的上端为\(R\),简单解释一下.
其中红色部分为已知部分的链,黑色部分为当前所求.
很容易发现,如果要拼接,我们需要将黑色部分左右端点调换,直接\(swap\)即可.
最后需要考虑的是当两端点在一条链上如何拼接,这里不再讨论,
请大家独立思考QwQ
本来应该一边切的,结果因为局部变量和重载出锅QAQ
代码
#include<cstdio>
#include<cstring>
#include<iostream>
#include<cctype>
#define ls o<<1
#define rs o<<1|1
#define R register
#define N 300008
using namespace std;
inline void in(int &x)
{
int f=1;x=0;char s=getchar();
while(!isdigit(s)){if(s=='-')f=-1;s=getchar();}
while(isdigit(s)){x=x*10+s-'0';s=getchar();}
x*=f;
}
struct cod{int lc,rc,sum;}tr[N<<2];
cod operator +( cod a, cod b)
{
cod res;
res.lc=a.lc;res.rc=b.rc;
res.sum=a.sum+b.sum;
if(a.rc==b.lc)res.sum--;
return res;
}
int n,m,head[N],tot,a[N],tg[N<<2];
struct code{int u,v;}edge[N<<2];
inline void add(int x,int y)
{
edge[++tot].u=head[x];
edge[tot].v=y;
head[x]=tot;
}
int size[N],son[N],f[N],depth[N];
void dfs1(int u,int fa)
{
depth[u]=depth[fa]+1;f[u]=fa;size[u]=1;
for(R int i=head[u];i;i=edge[i].u)
{
if(edge[i].v==fa)continue;
dfs1(edge[i].v,u);
size[u]+=size[edge[i].v];
if(son[u]==-1 or size[son[u]]<size[edge[i].v])
son[u]=edge[i].v;
}
}
int dfn[N],fdfn[N],idx,top[N];
void dfs2(int u,int t)
{
dfn[u]=++idx;fdfn[idx]=u;top[u]=t;
if(son[u]==-1)return;
dfs2(son[u],t);
for(R int i=head[u];i;i=edge[i].u)
{
if(dfn[edge[i].v])continue;
dfs2(edge[i].v,edge[i].v);
}
}
void build(int o,int l,int r)
{
if(l==r)
{
tr[o].lc=tr[o].rc=a[fdfn[l]];
tr[o].sum=1;
return;
}
int mid=(l+r)>>1;
build(ls,l,mid);
build(rs,mid+1,r);
tr[o]=tr[ls]+tr[rs];
}
inline void down(int o,int l,int r)
{
if(tg[o])
{
tg[ls]=tr[ls].lc=tr[ls].rc=tg[o];
tg[rs]=tr[rs].lc=tr[rs].rc=tg[o];
tr[ls].sum=tr[rs].sum=1;
tg[o]=0;
}
}
void change(int o,int l,int r,int x,int y,int z)
{
if(x<=l and y>=r)
{
tr[o].lc=tr[o].rc=tg[o]=z;
tr[o].sum=1;
return;
}
down(o,l,r);
int mid=(l+r)>>1;
if(x<=mid)change(ls,l,mid,x,y,z);
if(y>mid)change(rs,mid+1,r,x,y,z);
tr[o]=tr[ls]+tr[rs];
}
cod query(int o,int l,int r,int x,int y)
{
if(x<=l and y>=r)return tr[o];
down(o,l,r);
int mid=(l+r)>>1;
if(y<=mid)return query(ls,l,mid,x,y);
if(x>mid)return query(rs,mid+1,r,x,y);
return query(ls,l,mid,x,y)+query(rs,mid+1,r,x,y);
}
inline void tchange(int x,int y,int z)
{
int fx=top[x],fy=top[y];
while(fx!=fy)
{
if(depth[fx]>=depth[fy])
{
change(1,1,n,dfn[fx],dfn[x],z);
x=f[fx];
}
else
{
change(1,1,n,dfn[fy],dfn[y],z);
y=f[fy];
}
fx=top[x],fy=top[y];
}
if(dfn[x]>dfn[y])swap(x,y);
change(1,1,n,dfn[x],dfn[y],z);
}
inline cod tquery(int x,int y)
{
cod a,b,res;
a.lc=a.rc=b.lc=b.rc=a.sum=b.sum=0;
int fx=top[x],fy=top[y];
while(fx!=fy)
{
if(depth[fx]>=depth[fy])
{
res=query(1,1,n,dfn[fx],dfn[x]);
swap(res.lc,res.rc);
a=a+res;
x=f[fx];
}
else
{
res=query(1,1,n,dfn[fy],dfn[y]);
swap(res.lc,res.rc);
b=b+res;
y=f[fy];
}
fx=top[x],fy=top[y];
}
if(dfn[x]>dfn[y])
{
swap(x,y);
swap(a,b);
}
res=query(1,1,n,dfn[x],dfn[y]);
a=a+res;
swap(b.lc,b.rc);
return a+b;
}
char s[8];
int main()
{
in(n),in(m);memset(son,-1,sizeof son);
for(R int i=1;i<=n;i++)in(a[i]);
for(R int i=1,x,y;i<n;i++)
{
in(x),in(y);
add(x,y),add(y,x);
}
dfs1(1,0);dfs2(1,1);build(1,1,n);
for(R int a,b,c;m;m--)
{
scanf("%s",s+1);
if(s[1]=='Q')
{
in(a),in(b);
printf("%d\n",tquery(a,b).sum);
}
else
{
in(a),in(b),in(c);
tchange(a,b,c);
}
}
}
树链剖分【p2568】[SDOI2011]染色的更多相关文章
- BZOJ 2243: [SDOI2011]染色 [树链剖分]
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6651 Solved: 2432[Submit][Status ...
- bzoj-2243 2243: [SDOI2011]染色(树链剖分)
题目链接: 2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6267 Solved: 2291 Descript ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- bzoj 2243 [SDOI2011]染色(树链剖分,线段树)
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4637 Solved: 1726[Submit][Status ...
- Bzoj 2243: [SDOI2011]染色 树链剖分,LCT,动态树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 5020 Solved: 1872[Submit][Status ...
- bzoj 2243 [SDOI2011]染色(树链剖分+线段树合并)
[bzoj2243][SDOI2011]染色 2017年10月20日 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询 ...
- bzoj2243[SDOI2011]染色 树链剖分+线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 9012 Solved: 3375[Submit][Status ...
- B20J_2243_[SDOI2011]染色_树链剖分+线段树
B20J_2243_[SDOI2011]染色_树链剖分+线段树 一下午净调这题了,争取晚上多做几道. 题意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成 ...
- bzoj 2243: [SDOI2011]染色 线段树区间合并+树链剖分
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 7925 Solved: 2975[Submit][Status ...
- bzoj 2243: [SDOI2011]染色 (树链剖分+线段树 区间合并)
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 9854 Solved: 3725[Submit][Status ...
随机推荐
- jrebel插件激活
不管用的哪个工具都可以通过下面的教程获取注册码,激活你的开发工具 原文出自:http://www.gezila.com/tutorials/11476.html 首先打开Myeclipse,点击“he ...
- 关于控制下拉框select只读的js控制
文本框有readonly属性,直接设置:下拉框没有readonly属性,也不能通过其他属性进行只读的设置,下拉框只有disabled属性,但是这个属性设成true之后,值就获取不到了: 我在网上搜了一 ...
- 数据结构&图论:K短路-可持久化可并堆
本来A*就可以搞定的题,为了怕以后卡复杂度,找了个这么个方法 现阶段水平不够就不补充算法分析部分了 对于图G,建立一个以终点t为起点的最短路径构成的最短路径树 (就是反着跑一遍最短路,然后对于一个不为 ...
- 403 Forbidden 错误
Forbidden You don't have permission to access /a.php on this server. apache昨天调试 httpd.conf 文件:<Di ...
- abstract 与interface区别
1.abstract用于修饰类,interface用于修饰接口 2.抽象类中可以有抽象和非抽象方法,接口中只能定义抽象方法,不能有实现 3.抽象类必须被继承,interface被实现 4.抽象类有构造 ...
- 【hdu1251-统计难题】Trie
http://acm.hust.edu.cn/vjudge/problem/16379 题意:给定多个单词,多次询问符合某前缀的单词有多少个. 题解:tire.数组开了5*10^6才A,不然就RE. ...
- jzoj2701 【GDKOI2012模拟02.01】矩阵
传送门:https://jzoj.net/senior/#main/show/2701 [题目大意] 给出矩阵A,求矩阵B,使得
- php发送请求
$opts = array( 'http'=>array( 'method'=>"GET", 'timeout'=>10, ));$context = strea ...
- 智能合约安全-parity多重签名钱包安全漏洞
漏洞原因: 因为initWallet函数是公开函数( public function) , 攻击者调用initWallet,重新初始化钱包会把之前合约钱包所有者覆盖, 即可改变钱包所有者. 漏洞代码: ...
- 使用MapReduce将HDFS数据导入到HBase(三)
使用MapReduce生成HFile文件,通过BulkLoader方式(跳过WAL验证)批量加载到HBase表中 package com.mengyao.bigdata.hbase; import j ...