BZOJ 2243 染色 树链剖分
题意:
给出一棵树,每个顶点上有个颜色\(c_i\)。
有两种操作:
- C a b c 将\(a \to b\)的路径所有顶点上的颜色变为c
- Q a b 查询\(a \to b\)的路径上的颜色段数,连续相同颜色视为一段
分析:
首先树链剖分,下面考虑线段树部分:
我们维护一个区间的左端点的颜色和右断点的颜色以及该区间的颜色段数,在加一个颜色覆盖标记。
在pushup的时候,如果左区间右端点颜色和右区间左端点颜色相同,那么这段颜色可以合并,合并区间的颜色段数为左右子区间颜色段数之和减1;
否则,答案为左右子区间颜色段数之和。
本题的特殊性在于区间合并的顺序性,我们是自底向上将两个顶点跳到\(LCA\)的。因为在每条重链上,顶点在线段树上的编号是从上到下递增的。所以每个子查询得到的区间信息也是从上到下的。我们可以将所得区间左右翻转(具体就是交换区间左右端点颜色,颜色段数不会变)一下,再合并最终得到整个查询区间。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = 100000 + 10;
const int maxnode = maxn * 4;
struct Edge
{
int v, nxt;
Edge() {}
Edge(int v, int nxt):v(v), nxt(nxt) {}
};
int n, m, a[maxn];
int ecnt, head[maxn];
Edge edges[maxn * 2];
void AddEdge(int u, int v) {
edges[ecnt] = Edge(v, head[u]);
head[u] = ecnt++;
edges[ecnt] = Edge(u, head[v]);
head[v] = ecnt++;
}
int fa[maxn], dep[maxn], sz[maxn], son[maxn];
int tot, top[maxn], id[maxn], pos[maxn];
void dfs(int u) {
sz[u] = 1; son[u] = 0;
for(int i = head[u]; ~i; i = edges[i].nxt) {
int v = edges[i].v;
if(v == fa[u]) continue;
dep[v] = dep[u] + 1;
fa[v] = u;
dfs(v);
sz[u] += sz[v];
if(sz[v] > sz[son[u]]) son[u] = v;
}
}
void dfs2(int u, int tp) {
id[u] = ++tot;
pos[tot] = u;
top[u] = tp;
if(!son[u]) return;
dfs2(son[u], tp);
for(int i = head[u]; ~i; i = edges[i].nxt) {
int v = edges[i].v;
if(v == fa[u] || v == son[u]) continue;
dfs2(v, v);
}
}
struct Node
{
int lcol, rcol, cntv;
Node() {}
Node(int l, int r, int c): lcol(l), rcol(r), cntv(c) {}
};
void reverse(Node& t) { swap(t.lcol, t.rcol); }
Node operator + (const Node& a, const Node& b) {
if(!a.cntv) return b; if(!b.cntv) return a;
return Node(a.lcol, b.rcol, a.cntv + b.cntv - 1 + (a.rcol != b.lcol));
}
int setv[maxnode];
Node t[maxnode];
void build(int o, int L, int R) {
setv[o] = -1;
if(L == R) {
t[o].cntv = 1;
t[o].lcol = t[o].rcol = a[pos[L]];
return;
}
int M = (L + R) / 2;
build(o<<1, L, M);
build(o<<1|1, M+1, R);
t[o] = t[o<<1] + t[o<<1|1];
}
void pushdown(int o) {
if(setv[o] != -1) {
setv[o<<1] = setv[o<<1|1] = setv[o];
t[o<<1].cntv = t[o<<1|1].cntv = 1;
t[o<<1].lcol = t[o<<1].rcol = t[o<<1|1].lcol = t[o<<1|1].rcol = setv[o];
setv[o] = -1;
}
}
void update(int o, int L, int R, int qL, int qR, int v) {
if(qL <= L && R <= qR) {
t[o].lcol = t[o].rcol = setv[o] = v;
t[o].cntv = 1;
return;
}
pushdown(o);
int M = (L + R) / 2;
if(qL <= M) update(o<<1, L, M, qL, qR, v);
if(qR > M) update(o<<1|1, M+1, R, qL, qR, v);
t[o] = t[o<<1] + t[o<<1|1];
}
void UPDATE(int u, int v, int val) {
int t1 = top[u], t2 = top[v];
while(t1 != t2) {
if(dep[t1] < dep[t2]) { swap(u, v); swap(t1, t2); }
update(1, 1, n, id[t1], id[u], val);
u = fa[t1]; t1 = top[u];
}
if(dep[u] > dep[v]) swap(u, v);
update(1, 1, n, id[u], id[v], val);
}
Node query(int o, int L, int R, int qL, int qR) {
Node ans(0, 0, 0);
if(qL <= L && R <= qR) return t[o];
pushdown(o);
int M = (L + R) / 2;
if(qL <= M) ans = ans + query(o<<1, L, M, qL, qR);
if(qR > M) ans = ans + query(o<<1|1, M+1, R, qL, qR);
return ans;
}
int QUERY(int u, int v) {
Node q1(0, 0, 0), q2(0, 0, 0), tmp;
int t1 = top[u], t2 = top[v];
while(t1 != t2) {
if(dep[t1] > dep[t2]) {
tmp = query(1, 1, n, id[t1], id[u]);
reverse(tmp);
q1 = q1 + tmp;
u = fa[t1]; t1 = top[u];
} else {
tmp = query(1, 1, n, id[t2], id[v]);
reverse(tmp);
q2 = q2 + tmp;
v = fa[t2]; t2 = top[v];
}
}
if(dep[u] > dep[v]) {
tmp = query(1, 1, n, id[v], id[u]);
reverse(tmp);
q1 = q1 + tmp;
} else {
tmp = query(1, 1, n, id[u], id[v]);
reverse(tmp);
q2 = q2 + tmp;
}
reverse(q2);
q1 = q1 + q2;
return q1.cntv;
}
int main()
{
scanf("%d%d", &n, &m);
for(int i = 1; i <= n; i++) scanf("%d", a + i);
ecnt = 0;
memset(head, -1, sizeof(head));
for(int i = 1; i < n; i++) {
int u, v; scanf("%d%d", &u, &v);
AddEdge(u, v);
}
dfs(1);
tot = 0;
dfs2(1, 1);
build(1, 1, n);
char cmd[5];
int a, b, c;
while(m--) {
scanf("%s", cmd);
scanf("%d%d", &a, &b);
if(cmd[0] == 'C') {
scanf("%d", &c);
UPDATE(a, b, c);
} else {
printf("%d\n", QUERY(a, b));
}
}
return 0;
}
BZOJ 2243 染色 树链剖分的更多相关文章
- BZOJ 2243 染色 | 树链剖分模板题进阶版
BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上 ...
- BZOJ - 2243 染色 (树链剖分+线段树+区间合并)
题目链接 线段树维护区间连续段个数即可.设lc为区间左端点颜色,rc为区间右端点颜色,则合并两区间的时候,如果左区间右端点和右区间左端点颜色相同,则连续段个数-1. 在树链上的区间合并可以定义一个结构 ...
- hysbz 2243 染色(树链剖分)
题目链接:hysbz 2243 染色 题目大意:略. 解题思路:树链剖分+线段树的区间合并,可是区间合并比較简单,节点仅仅要记录左右端点的颜色就可以. #include <cstdio> ...
- HYSBZ - 2243 染色 (树链剖分+线段树)
题意:树上每个结点有自己的颜色,支持两种操作:1.将u到v路径上的点颜色修改为c; 2.求u到v路径上有多少段不同的颜色. 分析:树剖之后用线段树维护区间颜色段数.区间查询区间修改.线段树结点中维护的 ...
- BZOJ 2243: [SDOI2011]染色 [树链剖分]
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6651 Solved: 2432[Submit][Status ...
- Bzoj 2243: [SDOI2011]染色 树链剖分,LCT,动态树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 5020 Solved: 1872[Submit][Status ...
- BZOJ 2243: [SDOI2011]染色 树链剖分 倍增lca 线段树
2243: [SDOI2011]染色 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...
- BZOJ 2243: [SDOI2011]染色 树链剖分+线段树区间合并
2243: [SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数 ...
- BZOJ 2243: [SDOI2011]染色 (树链剖分+线段树合并)
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2243 树链剖分的点剖分+线段树.漏了一个小地方,调了一下午...... 还是要细心啊! 结 ...
随机推荐
- C#学习笔记:foreach原理
这篇随笔是对上一篇随笔C#关键字:yield的扩展. 关于foreach 首先,对于 foreach ,大家应该都非常熟悉,这里就简单的描述下. foreach 语句用于对实现 System.Col ...
- IE盒子模型和W3C盒子模型
IE盒模型出现在ie5.5以下的版本当中,ie6以上就实行W3C盒模型. box-sizing有两个属性,border-box和content-box. border-box对应传统的盒子模型,即ie ...
- SharePoint Server 2016 WEB 网站浏览器支持
SharePoint Server 2016支持多种常用的Web浏览器,如Internet Explorer,Google Chrome,Mozilla Firefox,Apple Safari和Mi ...
- selenium +python之多线程与多进程应用于自动化测试
多线程与多进程与自动化测试用例结合起来执行,从而节省测试用例的总体运行时间. 多线程执行测试测试用例 以百度搜索为例,通过不同的浏览器来启动不同的线程. from selenium import we ...
- Python+selenium 之操作Cookie
在验证浏览器中cookie是否正确时,有时基于真实cookie的测试是无法通过白盒和集成测试进行的.Webdriver提供了操作Cookie的相关方法,可以读取,添加和删除cookie信息. 文本we ...
- SQL server的一个分割表值函数
CREATE FUNCTION [dbo].[Fn_Split] ( @SplitString text, -- 如果要传入NText类型,下面需要相应的修改,注释行为NText下同 ) = ','- ...
- basic ,fundamental ,extreme ,utmost和radical.区别
basic 普通用词,指明确.具体的基础或起点.fundamental 书面用词,不如basic使用广泛,侧重指作为基础.根本的抽象的事物.radical 着重指事物的根本或其来源.下面是extrem ...
- HDU 3001 Travelling (状压DP,3进制)
题意: 给出n<=10个点,有m条边的无向图.问:可以从任意点出发,至多经过同一个点2次,遍历所有点的最小费用? 思路: 本题就是要卡你的内存,由于至多可经过同一个点2次,所以只能用3进制来表示 ...
- 【Python图像特征的音乐序列生成】如何标记照片的特征
目前我能想到的办法是这样的: 1,提取照片中的实体特征,借用某个pre-trained model进行tag标记. 2,将特征组合起来,形成一个bag-of-word model,然后将这个向量作为输 ...
- 恢复为TrustedInstaller权限
每次我们要改动系统文件/文件夹时,都会被提示权限不够,而这个文件的所有者就是TrustInstaller.所以,就出现各种各样的教程,甚至傻瓜式的一键操作,让大家把自己设为文件的所有者,让自己得到最高 ...