bzoj2243树链剖分+染色段数
终于做了一道不是一眼出思路的代码题(⊙o⊙)
之前没有接触过这种关于染色段数的题目(其实上课好像讲过),于是百度了一下(现在思维能力好弱)
实际上每一段有用的信息就是总共有几段和两段各是什么颜色,在开线段树的时候记录一下就好了
事实上我开了一个node,并且写了一个mix还是大大减小了代码量(对于我这种手残党来说同时大大减小了错误率)
由于前几题做的都是树链剖分,并没有在这一方面出问题,然而线段树还是不熟练,刚写完的时候居然忘记down了(mdzz)
对wa了N遍的总结:
由于是树上两个点间的路径,既有向上的路也有向下的路,然而树剖下来的结果是按深度排的,所以在链上跑的时候会发现一段链的左右(l和r)有时候是相反的,在合并几条链的时候一定要注意(事实证明我这么多遍wa全是因为没有翻转链)
对这一类问题的总结:
对于每一段都考虑有用(会对周围产生影响)的信息,在染色段数中就是两端(因为可能和隔壁合并)
//加了一大堆斜杠的是第一遍写之后补的
#include <cstdio>
#include <iostream>
#define mid (l+r)/2
using namespace std;
int m=,N=,n,M,p,q,o;
int to[],nex[],fir[],son[],bro[],co[];
int size[],l[],pos[],fa[],h[],top[];
struct node{int s,l,r;bool b;}t[],ans[];
inline void add(int x,int y){to[++m]=y;nex[m]=fir[x];fir[x]=m;}
inline node mix(node x,node y){return (node){x.s+y.s-(x.r==y.l),x.l,y.r,};}
int build(int now,int fat)
{
size[now]=;fa[now]=fat;h[now]=h[fat]+;
for(int i=fir[now];i;i=nex[i])
if(to[i]!=fat)
size[now]+=build(to[i],now),bro[to[i]]=son[now],son[now]=to[i];
return size[now];
}
void pou(int now,int to)
{
l[++N]=now;pos[now]=N;top[now]=to;
int max=son[now];
if(!max) return;
for(int i=bro[max];i;i=bro[i])
if(size[i]>size[max]) max=i;
pou(max,to);
for(int i=son[now];i;i=bro[i])
if(i!=max) pou(i,i);
}
void down(int now)////////////////////////////
{
if(t[now].b)
{
t[now].b=;t[now*].b=t[now*+].b=;
t[now*].s=t[now*+].s=;
t[now*].l=t[now*].r=t[now*+].l=t[now*+].r=t[now].l;
}
}
void work(int now,int l,int r,int x,int y,int z)
{
if(l==x && r==y)
{ t[now]=(node){,z,z,}; return;}
down(now);
if(x<=mid)
work(now*,l,mid,x,min(y,mid),z);
if(y>mid)
work(now*+,mid+,r,max(x,mid+),y,z);
t[now]=mix(t[now*],t[now*+]);
}
node que(int now,int l,int r,int x,int y)
{
if(l==x && r==y)
return t[now];
down(now);
if(y<=mid) return que(now*,l,mid,x,y);
if(x>mid) return que(now*+,mid+,r,x,y);
return mix(que(now*,l,mid,x,mid),que(now*+,mid+,r,mid+,y));
}
void solve(int x,int y,int z)
{
bool b=;ans[]=ans[]=(node){,-,-,};
while(top[x]!=top[y])
{
if(h[top[x]]<h[top[y]]) swap(x,y),b=!b;
if(z==-)
{
node tem=que(,,n,pos[top[x]],pos[x]);
if(!b) swap(tem.l,tem.r);///////////////////////////////
if(ans[b].s==) ans[b]=tem;
else
ans[b]=b?mix(tem,ans[b]):mix(ans[b],tem);
}
else
work(,,n,pos[top[x]],pos[x],z);
x=fa[top[x]];
}
if(h[x]>h[y]) swap(x,y),b=!b;//////////////////
if(z==-)
{
node tem=que(,,n,pos[x],pos[y]);b=!b;////////////////
if(!b) swap(tem.l,tem.r);
if(ans[b].s==) ans[b]=tem;
else
ans[b]=b?mix(tem,ans[b]):mix(ans[b],tem);
printf("%d\n",mix(ans[],ans[]).s);
}
else
work(,,n,pos[x],pos[y],z);
}
int main()
{
scanf("%d%d",&n,&M);
for(int i=;i<=n;i++)
scanf("%d",&co[i]);
for(int i=;i<n;i++)
scanf("%d%d",&p,&q),add(p,q),add(q,p);
build(,);
pou(,);
for(int i=;i<=n;i++)
work(,,n,pos[i],pos[i],co[i]);
for(int i=;i<=M;i++)
{
char ch=getchar();
while(ch!='C' && ch!='Q') ch=getchar();
if(ch=='C')
scanf("%d%d%d",&p,&q,&o),solve(p,q,o);
else
scanf("%d%d",&p,&q),solve(p,q,-);
}
return ;
}
bzoj2243树链剖分+染色段数的更多相关文章
- BZOJ2243 (树链剖分+线段树)
Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...
- bzoj2243 树链剖分
https://www.lydsy.com/JudgeOnline/problem.php?id=2243 新学的树剖,在维护的时候线段树维护区间内颜色数量以及左右两端的颜色.统计的时候区间合并时判断 ...
- bzoj2243树链剖分+区间合并
树链上区间合并的问题比区间修改要复杂,因为每一条重链在线段树上分布一般都是不连续的,所以在进行链上操作时要手动将其合并起来,维护两个端点值 处理时的方向问题:lca->u是一个方向,lca-&g ...
- [SDOI2011]染色 BZOJ2243 树链剖分+线段树
分析: 区间合并,lcol是左端点的颜色编号,rcol是右端点的颜色编号,那么我们向上合并的时候,如果左儿子的rcol等于右儿子的lcol那么区间的sum--. 另外,如果重链顶的颜色等于重链顶的父节 ...
- hdu3966 树链剖分+成段更新
给你n个点,m条边,p次操作.n个点相连后是一棵树.每次操作可以是x 到 y 增加 z,或者减z,或者问当前点的值是多少. 可以将树分成链,每个点在线段树上都有自己的点,然后线段树成段更新一下. #p ...
- BZOJ2243 [SDOI2011]染色(树链剖分+线段树合并)
题目链接 BZOJ2243 树链剖分 $+$ 线段树 线段树每个节点维护$lc$, $rc$, $s$ $lc$代表该区间的最左端的颜色,$rc$代表该区间的最右端的颜色 $s$代表该区间的所有连续颜 ...
- HDU3966-Aragorn's Story(树链剖分)
第一道树链剖分. 早就想学..一直懒.. 感觉还是比较简单的. 主要是要套其他数据结构,线段树大概还好,平衡树之类的肯定就跪了. http://blog.csdn.net/acdreamers/art ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- BZOJ2243 洛谷2486 [SDOI2011]染色 树链剖分
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2243 题目传送门 - 洛谷2486 题意概括 一棵树,共n个节点. 让你支持以下两种操作,共m次操 ...
随机推荐
- (转)C#根据当前时间获取周,月,季度,年度等时间段的起止时间
DateTime dt = DateTime.Now; //当前时间 DateTime startWeek = dt.AddDays( - Convert.ToInt32(dt.DayOfWeek.T ...
- (转)高效的将excel导入sqlserver中
大部分人都知道用oledb来读取数据到dataset,但是读取之后怎么处理dataset就千奇百怪了.很多人通过循环来拼接sql,这样做不但容易出错而且效率低下,System.Data.SqlClie ...
- grep 命令
简单介绍:grep命令是用于分析一行信息,若当中有我们所需要的信息,就将该行取出来. 语法结构:grep [-acinv] [--color=auto] '查找关键字' #{filename} -a: ...
- 419. Battleships in a Board
https://leetcode.com/problems/battleships-in-a-board/ 给定一个N×N的棋盘,有任意数量的1×N或N×1大小的"船",注意船船之 ...
- InstallSheild的一些常量
在用InstallShield制作安装包的时候,我们经常需要用到一些常量,弄清楚这些常量的具体含义,可以方便我们灵活使用脚本. TARGETDIR 默认安装路径,在安装过程中用户可手动更改.如:安装 ...
- [Sass]声明变量
[Sass]声明变量 定义变量的语法: 在有些编程语言中(如,JavaScript)声明变量都是使用关键词"var"开头,但是在 Sass 不使用这个关键词,而是使用大家都喜欢的美 ...
- java的异常总结
异常:在java程序中也出现不正常的情况,这个就叫异常.java是面向对象的语言.任何事物都可以用类来描述,同样异常也是一种事物,java中提供了很多异常类 很多异常堆积起来叫做异常体系 Throwa ...
- width:100%;与width:auto;的区别
<div> <p>1111</p> </div> div{ width:980px; background-color: #ccc; height:30 ...
- JAVA的模式对话框和非模式对话框
周末的时候,一位网友让我帮他把他的无模式对话框改成有模式对话框. 界面是由swing制作的,都是JFrame,我从来没有接触过swing编程.大致的代码还是看的懂,很多都和C#很相似. 然后就去查资料 ...
- 手把手教你玩转nginx负载均衡(一)----使用vitualBox创建虚拟机
引言 作为一个web程序员,有时候需要想尽办法来利用有限的资源来产生最大程度的负载,除了提高硬件配置,增加带宽之外,CDN加速,DNS加速,缓存,还可以利用反向代理.但是要说反向代理,就不的不说ngi ...