线段树区间合并就挺麻烦了,再套个树链就更加鬼畜,不过除了代码量大就没什么其他的了。。

一些细节:线段树每个结点用结构体保存,pushup等合并函数改成返回一个结构体,这样好写一些

struct Seg{
int lc,rc,tot;
Seg(){lc=rc=-;tot=;}
};
Seg seg[maxn<<];int lazy[maxn<<];
Seg pushup(Seg a,Seg b){
if(!a.tot)return b;
if(!b.tot)return a;
Seg res;
res.lc=a.lc,res.rc=b.rc;
res.tot=a.tot+b.tot;
if(a.rc==b.lc)res.tot--;
return res;
}

向上爬时更新操作不用变,但是询问操作需要改变

同样有一些值得注意的地方:向上爬的两条链是有顺序的,合并时顺序不能搞反,也不能像普通树链剖分那样直接swap

int Query(int x,int y){
Seg A,B;
while(top[x]!=top[y]){
if(d[top[x]]<d[top[y]]){
B=pushup(query(id[top[y]],id[y],,n,),B);
y=f[top[y]];
}
else {
A=pushup(query(id[top[x]],id[x],,n,),A);
x=f[top[x]];
}
}
if(id[x]>id[y])
A=pushup(query(id[y],id[x],,n,),A);
else
B=pushup(query(id[x],id[y],,n,),B);
if(A.lc==B.lc)return A.tot+B.tot-;
else return A.tot+B.tot;
}

最后是完整代码

#include<bits/stdc++.h>
using namespace std;
#define maxn 100005
struct Edge{int to,nxt;}edge[maxn<<];
int c[maxn],head[maxn],tot,n; int f[maxn],son[maxn],d[maxn],size[maxn];
int cnt,id[maxn],rk[maxn],top[maxn];
void dfs1(int x,int pre,int deep){
size[x]=,d[x]=deep;
for(int i=head[x];i!=-;i=edge[i].nxt){
int y=edge[i].to;
if(y==pre)continue;
f[y]=x;dfs1(y,x,deep+);size[x]+=size[y];
if(size[son[x]]<size[y])son[x]=y;
}
}
void dfs2(int x,int tp){
top[x]=tp;id[x]=++cnt;rk[cnt]=x;
if(son[x])dfs2(son[x],tp);
for(int i=head[x];i!=-;i=edge[i].nxt){
int y=edge[i].to;
if(y!=son[x] && y!=f[x])dfs2(y,y);
}
} #define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
struct Seg{
int lc,rc,tot;
Seg(){lc=rc=-;tot=;}
};
Seg seg[maxn<<];int lazy[maxn<<];
Seg pushup(Seg a,Seg b){
if(!a.tot)return b;
if(!b.tot)return a;
Seg res;
res.lc=a.lc,res.rc=b.rc;
res.tot=a.tot+b.tot;
if(a.rc==b.lc)res.tot--;
return res;
}
void pushdown(int rt){
if(lazy[rt]<)return;
lazy[rt<<]=lazy[rt<<|]=lazy[rt];
seg[rt<<].lc=seg[rt<<].rc=lazy[rt];
seg[rt<<].tot=;
seg[rt<<|].lc=seg[rt<<|].rc=lazy[rt];
seg[rt<<|].tot=;
lazy[rt]=-;
}
void build(int l,int r,int rt){
if(l==r){
seg[rt].lc=seg[rt].rc=c[rk[l]];
seg[rt].tot=;return;
}
int m=l+r>>;
build(lson);build(rson);
seg[rt]=pushup(seg[rt<<],seg[rt<<|]);
}
void update(int L,int R,int c,int l,int r,int rt){
if(L<=l && R>=r){
lazy[rt]=c;seg[rt].lc=seg[rt].rc=c;
seg[rt].tot=;return;
}
pushdown(rt);
int m=l+r>>;
if(L<=m)update(L,R,c,lson);
if(R>m)update(L,R,c,rson);
seg[rt]=pushup(seg[rt<<],seg[rt<<|]);
}
Seg query(int L,int R,int l,int r,int rt){
if(L<=l && R>=r)return seg[rt];
pushdown(rt);
int m=l+r>>;
Seg res;
if(L<=m)res=pushup(res,query(L,R,lson));
if(R>m)res=pushup(res,query(L,R,rson));
return res;
} void Update(int x,int y,int c){
while(top[x]!=top[y]){
if(d[top[x]]<d[top[y]])swap(x,y);
update(id[top[x]],id[x],c,,n,);
x=f[top[x]];
}
if(id[x]>id[y])swap(x,y);
update(id[x],id[y],c,,n,);
}
int Query(int x,int y){
Seg A,B;
while(top[x]!=top[y]){
if(d[top[x]]<d[top[y]]){
B=pushup(query(id[top[y]],id[y],,n,),B);
y=f[top[y]];
}
else {
A=pushup(query(id[top[x]],id[x],,n,),A);
x=f[top[x]];
}
}
if(id[x]>id[y])
A=pushup(query(id[y],id[x],,n,),A);
else
B=pushup(query(id[x],id[y],,n,),B);
if(A.lc==B.lc)return A.tot+B.tot-;
else return A.tot+B.tot;
} void init(){
memset(head,-,sizeof head);
memset(lazy,-,sizeof lazy);
tot=;
}
void addedge(int u,int v){
edge[tot].to=v;edge[tot].nxt=head[u];head[u]=tot++;
}
int main(){
init();int q;
cin>>n>>q;
for(int i=;i<=n;i++)cin>>c[i];
for(int i=;i<n;i++){
int x,y;cin>>x>>y;
addedge(x,y);addedge(y,x);
}
cnt=;dfs1(,,),dfs2(,);
build(,n,);
char op[];int x,y,z;
while(q--){
scanf("%s",op);
if(op[]=='Q'){scanf("%d%d",&x,&y);
cout<<Query(x,y)<<'\n';}
if(op[]=='C'){scanf("%d%d%d",&x,&y,&z);Update(x,y,z);}
}
}

树链剖分——线段树区间合并bzoj染色的更多相关文章

  1. 【bzoj2325】[ZJOI2011]道馆之战 树链剖分+线段树区间合并

    题目描述 给定一棵树,每个节点有上下两个格子,每个格子的状态为能走或不能走.m次操作,每次修改一个节点的状态,或询问:把一条路径上的所有格子拼起来形成一个宽度为2的长方形,从起点端两个格子的任意一个开 ...

  2. BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)

    前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...

  3. bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)

    4196: [Noi2015]软件包管理器 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 2852  Solved: 1668[Submit][Sta ...

  4. BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )

    BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...

  5. BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)

    BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...

  6. POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 )

    POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 ) 题意分析 给出n个点,m个询问,和当前位置pos. 先给出n-1条边,u->v以及边权w. 然后有m个询问 ...

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

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

  8. BZOJ 3589 动态树 (树链剖分+线段树)

    前言 众所周知,90%90\%90%的题目与解法毫无关系. 题意 有一棵有根树,两种操作.一种是子树内每一个点的权值加上一个同一个数,另一种是查询多条路径的并的点权之和. 分析 很容易看出是树链剖分+ ...

  9. bzoj 2157: 旅游【树链剖分+线段树】

    裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...

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

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

随机推荐

  1. YApi二次开发环境部署

    YApi 是去哪儿网大前端技术中心开源的一个可视化的接口管理平台.下面是在CentOS7下搭建二次开发环境步骤. 1 安装MongoDB 具体安装参考CentOS7安装MongoDB 2 安装Node ...

  2. Hadoop系列(一):Hadoop集群搭建

    环境:CentOS 7 JDK: 1.7.0_80 hadoop:2.8.5 两台机器:master(192.168.56.101)   slave(192.168.56.102) 配置基础环境 1. ...

  3. bash中打印文件每一行及其行号

    #!/bin/bash linenumber=$(cat remoteIP.cfg |wc -l) currentline= for ip in $(cat remoteIP.cfg) do curr ...

  4. 如何对mRemoteNG在进行Linux终端访问时自定义配色

    Its not that easy to config mRemoteNG custom color themes, not like XShell which is really convinent ...

  5. css控制table间距

    border-collapse: separate;border-spacing: 20px 30px; /* 水平间距 垂直间距 */

  6. flink部署操作-flink standalone集群安装部署

    flink集群安装部署 standalone集群模式 必须依赖 必须的软件 JAVA_HOME配置 flink安装 配置flink 启动flink 添加Jobmanager/taskmanager 实 ...

  7. libiconv交叉编译提示arm-none-linux-gnueabi-gcc

    title: libiconv交叉编译提示arm-none-linux-gnueabi-gcc date: 2019/3/6 17:45:48 toc: true --- libiconv交叉编译提示 ...

  8. react项目后台及上线步骤

    应同学要求,本人将react项目创建后台及上线流程书写如下: 前端部分 略…… 后台部分 (注:这里的后台是用的nodejs搭建的,使用的是express框架+ejs模板) 首先通过express快速 ...

  9. 20155324 实验5 MSF基础应用

    20155324 实验5 MSF基础应用 ms08_067 用search命令,搜索与ms08_067相关的模块,如图: 服务器信息块(SMB)是一个网络文件共享协议,它允许应用程序和终端用户从远端的 ...

  10. python flsak 框架

    1.flask  轻量级微型web框架 优点:微框架.简单.可扩展 将flask变量实例化到app变量中 如果想要flask自动加载修改后的代码,要app.run(debug=True) 2.路由和视 ...