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$路径上的颜色段数量(连 ...
随机推荐
- linux下定时器介绍2--timer_create等函数集的使用示例
程序1:采用新线程派驻的通知方式 程序2:通知方式为信号的处理方式 #include <stdio.h>#include <time.h>#include <stdlib ...
- aarch64_m1
MAKEDEV-3.24-18.fc26.aarch64.rpm 2017-02-14 08:46 99K fedora Mirroring Project MUMPS-5.0.2-8.fc26.aa ...
- 使用脚本实现killproc的功能
在shell提示符号下输入type killproc,会发现killproc实在 /sbin/目录下,通过man killproc可以查看这个脚本(姑且这么称为脚本)的用法,现在,把这个脚本的实现过程 ...
- redis从入门到放弃 -> 部署方案
单点部署方案 环境准备: [root@localhost ~]# cat /etc/redhat-release CentOS Linux release 7.2.1511 (Core) [root@ ...
- spring学习之三 数据库操作jdbcTemplate
概念 jdbcTemplate就Spring对数据库持久化技术的实现,通过它可以对数据库进行CRUD等操作. JDBCTemplate和代码实现 public void jdbcadd() { Dri ...
- python网络编程--线程event
一:线程event作用 Python提供了Event对象用于线程间通信,它是线程设置的信号标志,如果信号标志位真,则其他线程等待直到信号结束. Event对象实现了简单的线程通信机制,它提供了设置信号 ...
- 浮动元素垂直居中,bootstrap栅格布局垂直居中
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- CF 248B 前缀和
操作1 l r 是原序列l-r的和操作2 l r 是从小到大排序后的l-r的和 input66 4 2 7 2 732 3 61 3 41 1 6output24928 # include <i ...
- Hive(四)Hive的3种连接方式与DbVisualizer连接Hive
一.CLI连接 进入到 bin 目录下,直接输入命令: [root@node21 ~]# hive SLF4J: Class path contains multiple SLF4J bindings ...
- CentOS7.5搭建ELK6.2.4集群及插件安装
一 简介 Elasticsearch是一个高度可扩展的开源全文搜索和分析引擎.它允许您快速,近实时地存储,搜索和分析大量数据.它通常用作支持具有复杂搜索功能和需求的应用程序的底层引擎/技术. 下载地址 ...