原题链接


肯定是树链剖分的题啦

树剖怎么做可以看我上一篇博客

如果我们已经剖完了:

然后考虑怎么维护重链和查询

用线段树维护的时候当前区间的区间颜色个数应该等于左儿子+右儿子,但是当左儿子的右端点和右儿子的左端点颜色一样,显然区间数要减1

所以每个节点存一下左端点的右端点颜色正常维护即可

考虑查询,在同一重链上的点显然线段树可以解决,当top[u]!=top[v]的时候,得让deep较深的爬树,那么答案就要+=爬树那一段的区间个数

但是有可能的是fa[top[u]]的颜色和top[u]的颜色相等,这个时候需要答案--,这样才能保证下次爬树的时候不会多算

注意:请写单点查询询问颜色,因为fa[top[u]]和top[u]是轻链的两端,不能保证在线段树中编号连续,所以不能区间查询是否是个数是1

修改同理,爬树的时候修改即可

 #include<cstdio>
#include<algorithm>
#include<cstring>
#define N 100010
using namespace std;
int n,m,head[N],indx[N],pos[N],fa[N],ecnt,color[N],deep[N],sz[N],top[N],a,b,c,tot,son[N];
char s[N];
int read()
{
int ret=,neg=;
char j=getchar();
for (;j>'' || j<'';j=getchar())
if (j == '-') neg=-;
for (;j>='' && j<='';j=getchar())
ret=ret*+j-'';
return ret*neg;
}
struct adj
{
int nxt,v;
}e[*N];
struct node
{
int l,r,Lcolor,Rcolor,sum,lz;
}t[*N];
void add(int u,int v)//加边
{
e[++ecnt].v=v;
e[ecnt].nxt=head[u];
head[u]=ecnt;
e[++ecnt].v=u;
e[ecnt].nxt=head[v];
head[v]=ecnt;
}
void dfs1(int x,int father,int d)//第一次dfs
{
fa[x]=father,deep[x]=d,sz[x]=;
for (int i=head[x];i;i=e[i].nxt)
{
int v=e[i].v;
if (v==father) continue;
dfs1(v,x,d+);
sz[x]+=sz[v];
if (sz[son[x]]<sz[v]) son[x]=v;
}
}
void dfs2(int x,int TOP)//第二次dfs
{
pos[x]=++tot;
indx[tot]=x;
top[x]=TOP;
if (son[x]) dfs2(son[x],TOP);
for (int i=head[x];i;i=e[i].nxt)
{
int v=e[i].v;
if (v==fa[x] || v==son[x]) continue;
if (v!=) dfs2(v,v);
}
}
void pushup(int p)//线段树更新
{
t[p].sum=t[p<<].sum+t[p<<|].sum;
if (t[p<<].Rcolor==t[p<<|].Lcolor) t[p].sum--;
t[p].Lcolor=t[p<<].Lcolor;
t[p].Rcolor=t[p<<|].Rcolor;
}
void pushdown(int p)//lazy下放
{
if (t[p].l==t[p].r || t[p].lz==-) return;
int w=t[p].lz;
t[p<<].Lcolor=t[p<<].Rcolor=t[p<<|].Lcolor=t[p<<|].Rcolor=t[p<<].lz=t[p<<|].lz=w;
t[p<<].sum=t[p<<|].sum=;
t[p].lz=-;
}
void build(int p,int l,int r)//建树
{
t[p].l=l,t[p].r=r,t[p].lz=-;
if (l==r)
{
t[p].Lcolor=t[p].Rcolor=color[indx[l]];
t[p].sum=;
}
else
{
int mid=l+r>>;
build(p<<,l,mid);
build(p<<|,mid+,r);
pushup(p);
}
}
void modify(int p,int l,int r,int k)//区间修改
{
if (l==t[p].l && r==t[p].r)
{
t[p].sum=;
t[p].Lcolor=t[p].Rcolor=k;
t[p].lz=k;
return;
}
pushdown(p);
int mid=t[p].l+t[p].r>>;
if (r<=mid)
modify(p<<,l,r,k);
else if (l>mid) modify(p<<|,l,r,k);
else
modify(p<<,l,mid,k),modify(p<<|,mid+,r,k);
pushup(p);
}
int query(int p,int l,int r)//区间询问
{
if (l==t[p].l && r==t[p].r)
return t[p].sum;
pushdown(p);
int mid=t[p].l+t[p].r>>;
if (r<=mid)
return query(p<<,l,r);
if (l>mid) return query(p<<|,l,r);
int tmp=query(p<<,l,mid)+query(p<<|,mid+,r);
if (t[p<<].Rcolor==t[p<<|].Lcolor) tmp--;
return tmp;
}
int Qcolor(int p,int l)//单点询问
{
if (t[p].l==t[p].r && t[p].l==l) return t[p].Lcolor;
pushdown(p);
int mid=t[p].l+t[p].r>>;
if (l<=mid) return Qcolor(p<<,l);
else return Qcolor(p<<|,l);
}
void getcolor(int a,int b,int w)//修改
{
while (top[a]!=top[b])//爬树
{
if (deep[top[a]]<deep[top[b]]) swap(a,b);
modify(,pos[top[a]],pos[a],w);
a=fa[top[a]];
}
if (deep[a]>deep[b]) swap(a,b);
modify(,pos[a],pos[b],w);
}
int querycolor(int u,int v)//询问
{
int ret=;
while (top[u]!=top[v])
{
if (deep[top[u]]<deep[top[v]]) swap(u,v);
ret+=query(,pos[top[u]],pos[u]);
if (Qcolor(,pos[fa[top[u]]])==Qcolor(,pos[top[u]])) ret--;//看题解
u=fa[top[u]];
}
if (deep[u]>deep[v]) swap(u,v);
return ret+query(,pos[u],pos[v]);
}
int main()
{
n=read(),m=read();
for (int i=;i<=n;i++)
color[i]=read();
for (int i=;i<n;i++)
add(read(),read());
dfs1(,,);
dfs2(,);
build(,,n);
while (m--)
{
scanf("%s",s);
if (s[]=='C')
{
a=read(),b=read(),c=read();
getcolor(a,b,c);
}
else
{
scanf("%d%d",&a,&b);
printf("%d\n",querycolor(a,b));
}
}
return ;
}

POJ 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. POJ.2763 Housewife Wind ( 边权树链剖分 线段树维护区间和 )

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

  7. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  8. 【bzoj1959】[Ahoi2005]LANE 航线规划 树链剖分+线段树

    题目描述 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系——一个巨大的由千百万星球构成的Samuel星系. 星际空间站的Samuel II巨型计算 ...

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

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

  10. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

随机推荐

  1. call()和apply()

    每个函数都包含apply和call方法. 相同点:都接收两个参数,一个是在其中运行函数的作用域,另一个是参数: 不同点:call方法和apply的不同之处在于接收参数的方式不同: apply方法第二个 ...

  2. php-5.6.26源代码 - 扩展模块的种类,扩展模块的执行埋点

    模块种类(两种) 类型一:zend的模块:(类似zend_extension=test.so) 识别方法: php.ini中以zend_extension开头的配置,如zend_extension=t ...

  3. PHP小练习题

    前几天在百度知道里面看到有位网友询问如何制作一下的小程序:用php语言设计一个小程序,计算今天到达下月的天数.全部输出这些天数,并使得每天的日期以三种颜色循环显示,设置三个表单,让用户选择字体颜色,然 ...

  4. http虚拟主机的简单配置训练

    http的虚拟主机 对于某些web访问站点而言,每天的访问量很少,因此真正的放一台服务器去进行web站点是很 浪费资源的,因此我们选择了虚拟主机 web处理模块的分类(MPM) 1.perfork 一 ...

  5. Triangular Sums 南阳acm122

    Triangular Sums 时间限制:3000 ms  |  内存限制:65535 KB 难度:2   描述 The nth Triangular number, T(n) = 1 + … + n ...

  6. linux-shell——01

    没有什么好的标题,只是一些随笔.我用的是linux虚拟机,red hat 7 一:nat模式使得虚拟机可以访问外网,但是这种模式下只可以访问外网,但外面的不能访问里面 首先将虚拟机的网络连接改为nat ...

  7. 操作视频-对视频进行canny边缘检测

    #include<opencv2/opencv.hpp> using namespace cv; int main() { VideoCapture capture(); //从摄像头读入 ...

  8. [BZOJ2243][SDOI2011]染色(树链剖分)

    [传送门] 树链剖分就行了,注意线段树上颜色的合并 Code #include <cstdio> #include <algorithm> #define N 100010 # ...

  9. Spring---BeanFactory与ApplicationContext简介

    BeanFactory概念 Spring通过一个配置文件描述bean和bean之间的依赖关系,然后利用java语言的反射功能实例化bean,并建立bean之间的依赖关系.Spring的IOC容器在完成 ...

  10. 《算法》C++代码 Dijkstra

    单源最短路,复杂度是O(N²),堆优化的是O(NlogN).基本思想是贪心,每次都加入一个当前最近的点,可以证明每次当时最近的点就是当前最短的路径.因此,所有点都加入之后,起点到所有点的最短路径就都求 ...