bzoj2243 染色
Description
Input
Output
对于每个询问操作,输出一行答案。
Sample Input
2 2 1 2 1 1
1 2
1 3
2 4
2 5
2 6
Q 3 5
C 2 1 1
Q 3 5
C 5 1 2
Q 3 5
Sample Output
1
2
HINT
数N<=10^5,操作数M<=10^5,所有的颜色C为整数且在[0, 10^9]之间。
第一眼看上去肯定是树链剖分,然后就是想怎么用线段树维护区间色段。
我们用线段树维护一个区间最左边的颜色,最右边的颜色,和颜色段数。如果一个节点的左儿子的右颜色和右儿子的左颜色相同,那么它的色段数是左+右-1,否则是左+右。
但是在查询时一定要注意,跑完每一条重链,和下一条重链中的轻链时,他们在线段树上并不是一起查询的。我们需要单点找出当前重链的顶端和下一个重链的底端的颜色,如果颜色相同,那么ans-1.
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstdlib>
#include <cstring>
#define in(a) a=read()
#define REP(i,k,n) for(int i=k;i<=n;i++)
#define MAXN 100010
using namespace std;
inline int read(){
int x=,f=;
char ch=getchar();
for(;!isdigit(ch);ch=getchar())
if(ch=='-')
f=-;
for(;isdigit(ch);ch=getchar())
x=x*+ch-'';
return x*f;
}
int n,m,a,b,d;
char c;
int input[MAXN];
int total,head[MAXN],nxt[MAXN<<],to[MAXN<<];
int depth[MAXN],size[MAXN],son[MAXN],f[MAXN];
int cnt,dfn[MAXN],top[MAXN],link[MAXN];
struct node{
int l,r,lc,rc,s,lt;
}tree[MAXN<<];
inline void adl(int a,int b){
total++;
to[total]=b;
nxt[total]=head[a];
head[a]=total;
return ;
}
inline void getson(int u,int fa){//得到重儿子
size[u]=;
for(int e=head[u];e;e=nxt[e])
if(to[e]!=fa){
depth[to[e]]=depth[u]+;
f[to[e]]=u;
getson(to[e],u);
size[u]+=size[to[e]];
if(!son[u] || size[to[e]]>size[son[u]]) son[u]=to[e];
}
return ;
}
inline void getdfn(int u,int t){//得到重边
top[u]=t;
dfn[u]=++cnt;
link[cnt]=u;
if(!son[u]) return ;
getdfn(son[u],t);
for(int e=head[u];e;e=nxt[e])
if(to[e]!=f[u] && to[e]!=son[u])
getdfn(to[e],to[e]);
return ;
}
inline void build(int i,int l,int r){//建树
tree[i].l=l;
tree[i].r=r;
if(l==r){
tree[i].s=,tree[i].lc=tree[i].rc=input[link[l]];
return ;
}
int mid=(l+r)>>;
build(i<<,l,mid);
build(i<<|,mid+,r);
if(tree[i<<].rc==tree[i<<|].lc) tree[i].s=tree[i<<].s+tree[i<<|].s-;
else tree[i].s=tree[i<<].s+tree[i<<|].s;
tree[i].lc=tree[i<<].lc;
tree[i].rc=tree[i<<|].rc;
}
inline void pushdown(int i){//下传懒标记
if(!tree[i].lt) return ;
int k=tree[i].lt;
tree[i<<].s=tree[i<<|].s=;
tree[i<<].lc=tree[i<<].rc=tree[i<<|].lc=tree[i<<|].rc=k;
tree[i<<].lt=tree[i<<|].lt=k;
tree[i].lt=;
return ;
}
inline void add(int i,int l,int r,int k){//修改颜色
if(tree[i].l>=l && tree[i].r<=r){
tree[i].s=;
tree[i].lt=tree[i].lc=tree[i].rc=k;
return ;
}
pushdown(i);
if(tree[i<<].r>=l) add(i<<,l,r,k);
if(tree[i<<|].l<=r) add(i<<|,l,r,k);
if(tree[i<<].rc==tree[i<<|].lc) tree[i].s=tree[i<<].s+tree[i<<|].s-;
else tree[i].s=tree[i<<].s+tree[i<<|].s;
tree[i].lc=tree[i<<].lc;
tree[i].rc=tree[i<<|].rc;
return ;
}
inline void updates(int x,int y,int z){//枚举两点间每一条重边
int tx=top[x],ty=top[y];
while(tx!=ty){
if(depth[tx]<depth[ty]) swap(tx,ty),swap(x,y);
add(,dfn[tx],dfn[x],z);
x=f[tx];
tx=top[x],ty=top[y];
}
if(depth[x]<depth[y]) swap(x,y);
add(,dfn[y],dfn[x],z);
}
inline int query(int i,int l,int r){//区间查询
int sum=;
if(tree[i].l>=l && tree[i].r<=r) return tree[i].s;
pushdown(i);
if(tree[i<<].r>=l) sum+=query(i<<,l,r);
if(tree[i<<|].l<=r) sum+=query(i<<|,l,r);
if(tree[i<<].r>=l && tree[i<<|].l<=r && tree[i<<].rc==tree[i<<|].lc) sum--;
return sum;
}
inline int getcolor(int i,int dis){//查询单点颜色
if(tree[i].l==tree[i].r) return tree[i].lc;
pushdown(i);
int mid=(tree[i].l+tree[i].r)>>;
if(dis<=mid) return getcolor(i<<,dis);
else return getcolor(i<<|,dis);
}
inline int getsum(int x,int y){//枚举查询时两点间的重边
int tx=top[x],ty=top[y],ans=;
while(tx!=ty){
if(depth[tx]<depth[ty]) swap(tx,ty),swap(x,y);
ans+=query(,dfn[tx],dfn[x]);
if(getcolor(,dfn[tx])==getcolor(,dfn[f[tx]])) ans--;//看轻边两点的颜色是否相同
x=f[tx];
tx=top[x],ty=top[y];
}
if(depth[x]<depth[y]) swap(x,y);
ans+=query(,dfn[y],dfn[x]);
return ans;
}
int main(){
in(n),in(m);
REP(i,,n) in(input[i]);
REP(i,,n-) in(a),in(b),adl(a,b),adl(b,a);
depth[]=;
getson(,);
getdfn(,);
build(,,n);
REP(i,,m){
cin>>c;
if(c=='C') in(a),in(b),in(d),updates(a,b,d);
if(c=='Q') in(a),in(b),printf("%d\n",getsum(a,b));
}
}
bzoj2243 染色的更多相关文章
- 刷题总结——bzoj2243染色
题目: 题目背景 SDOI2011 DAY1 T3 题目描述 给定一棵有 n 个节点的无根树和 m 个操作,操作有 2 类:1.将节点 a 到节点 b 路径上所有点都染成颜色 c :2.询问节点 a ...
- HDU5892~HDU5901 2016网络赛沈阳
A.题意: 有一个n×n的格子, 有50种怪物. 有m个操作, 每次操作会往一个矩形区域放怪物, 每个格子放相同数目的怪物, 或者查询当前50种怪物的奇偶性. 分析:用2^50表示怪物的奇偶,然后就是 ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
- BZOJ2243 洛谷2486 [SDOI2011]染色 树链剖分
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2243 题目传送门 - 洛谷2486 题意概括 一棵树,共n个节点. 让你支持以下两种操作,共m次操 ...
- BZOJ2243 SDOI2011 染色 【树链剖分】
BZOJ2243 SDOI2011 染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色 ...
- 【BZOJ2243】染色(树链剖分)
题意: 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如“112221”由 ...
- bzoj2243树链剖分+染色段数
终于做了一道不是一眼出思路的代码题(⊙o⊙) 之前没有接触过这种关于染色段数的题目(其实上课好像讲过),于是百度了一下(现在思维能力好弱) 实际上每一段有用的信息就是总共有几段和两段各是什么颜色,在开 ...
- bzoj-2243 2243: [SDOI2011]染色(树链剖分)
题目链接: 2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6267 Solved: 2291 Descript ...
- [bzoj2243][SDOI2011]染色
Description 给定一棵有$n$个节点的无根树和$m$个操作,操作有$2$类: 1.将节点$a$到节点$b$路径上所有点都染成颜色$c$; 2.询问节点$a$到节点$b$路径上的颜色段数量(连 ...
随机推荐
- XSS注入常用语句
<script>alert('hello,gaga!');</script> //经典语句,哈哈! >"'><img src="javas ...
- React-Native 之 FlexBox介绍和使用
# 前言 学习本系列内容需要具备一定 HTML 开发基础,没有基础的朋友可以先转至 HTML快速入门(一) 学习 本人接触 React Native 时间并不是特别长,所以对其中的内容和性质了解可能会 ...
- Web测试技术要领
基于Web的系统测试与传统的软件测试既有相同之处,也有不同的地方,对软件测试提出了新的挑战.基于Web的系统测试不但需要检查和验证是否按照设计的要求运行,而且还要评价系统在不同用户的浏览器端的显示是否 ...
- ROS + Caffe 机器人操作系统框架和深度学习框架笔记 (機器人控制與人工智能)
ROS + Caffe,这里以环境中物体识别为示例,机器人怎么知道环境里面有什么呢? [0.0567392 - n03376595 folding chair][0.0566773 - n040999 ...
- 第一篇CodeIgniter框架的下载及安装
初次学习Php,网上搜了很多php框架,最后选择了CodeIgniter. 安装环境:php5+mysql6.5+iis7 我的电脑是用来办公写文档用的,win7系统,不想换系统,所以就安装了win7 ...
- sql server2014 企业版 百度云下载
sql server2014 企业版 百度云下载 链接: https://pan.baidu.com/s/1j7a6RWwpvSzG-sF7Dnexfw 提取码: 关注公众号[GitHubCN]回复获 ...
- android发布版本的几个命令
./build_native.sh /opt/software/apache-ant-1.8.2/bin/ant clean #/opt/software/apache-ant-1.8.2/bin/a ...
- CF475C. Kamal-ol-molk's Painting
C. Kamal-ol-molk's Painting time limit per test 2 seconds memory limit per test 256 megabytes input ...
- linux 101 hacks 5PS1
PS1——默认提示符 看完这一章,我心里若干个卧槽.. 如下所示, 可以通过修改 Linux 下的默认提示符,使其更加实用.在下面的例子中,默认的 PS1的值是“ \s-\v\$”,显示出了 shel ...
- 【LOJ】#2443. 「NOI2011」智能车比赛
题解 显然是个\(n^2\)的dp 我们要找每个点不穿过非赛道区域能到达哪些区域的交点 可以通过控制两条向量负责最靠下的上边界,和最靠上的下边界,检查当前点在不在这两条向量之间即可,对于每个点可以\( ...