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

一些细节:线段树每个结点用结构体保存,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. Oracle 给予访问其他用户包的权限

    grant execute on apps.SPM_CON_INVOICE_INF_PKG to diq; grant  DEBUG on apps.SPM_CON_INVOICE_INF_PKG t ...

  2. js 替换所有指定的字符串

    js 的replace方法只替换第一个匹配到的的字符 如果要全局替换,使用以下方法(g为全局标志) str.replace(/需要替换的字符串/g,"新字符串") //如果有特殊符 ...

  3. python第七天

    复习: 1.深浅拷贝 值拷贝:直接赋值 = 号, 列表中的任何值发生改变,第二个中的值都会随之改变浅拷贝:通过copy()方法 ls2 = ls.copy(),第一个中存放的值的地址没有改变, 但内部 ...

  4. 年月日时分秒毫秒+随机数getSerialNum

    package com.creditharmony.apporveadapter.core.utils; import java.io.ByteArrayInputStream; import jav ...

  5. python解决上楼梯问题

    假设一段楼梯共n(n>1)个台阶,小朋友一步最多能上3个台阶,那么小朋友上这段楼梯一共有多少种方法 (此为京东2016年笔试题目) 假设n为15,从第15个台阶上往回看,有3种方法可以上来(从第 ...

  6. Python 中使用 matplotlib 绘图中文字符显示异常的问题

    最近在使用 Python matplotlib 绘制图表时发现中文字符不能正确显示:比如在绘制折线图时,中文全部显示成▢▢▢的格式,虽然将数据改成英文就没什么问题,但是所有数据都这么做时不可行的,于是 ...

  7. sed命令(二)

    转自:https://www.cnblogs.com/maxincai/p/5146338.html sed命令用法 sed是一种流编辑器,它是文本处理中非常有用的工具,能够完美的配合正则表达式使用, ...

  8. python 生产者 --- 消费者

    值得拿出来 看看的 多进程 爬取 (生产) , 解析 (消费) 网页 同时进行,可以作为以后项目扩展使用 from bs4 import BeautifulSoup import requests i ...

  9. OSPFv3实验配置(GNS3)

    实验目的 1. 掌握 OSPFv3(v2) 的配置方法 2. 掌握在帧中继环境下 OSPFv3 (v2)的配置方法 3. 掌握 OSPFv3(v2) NSSA 的配置方法 4. 掌握外部路由汇总的配置 ...

  10. 如何运用jieba库分词

    使用jieba库分词 一.什么是jieba库 1.jieba库概述 jieba是优秀的中文分词第三方库,中文文本需要通过分词获得单个词语. 2.jieba库的使用:(jieba库支持3种分词模式) 通 ...