BZOJ 2243 染色 | 树链剖分模板题进阶版
BZOJ 2243 染色 | 树链剖分模板题进阶版
这道题呢就是个带区间修改的树链剖分
如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了。
这道题要注意的是,无论是线段树上还是原树上,把两个区间的信息合并的时候,要注意中间相邻两个颜色是否相同。
这代码好长啊啊啊啊
幸好一次过了不然我估计永远也De不出来
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
#define space putchar(' ')
#define enter putchar('\n')
template <class T>
void read(T &x){
char c;
bool op = 0;
while(c = getchar(), c < '0' || c > '9')
if(c == '-') op = 1;
x = c - '0';
while(c = getchar(), c >= '0' && c <= '9')
x = x * 10 + c - '0';
if(op) x = -x;
}
template <class T>
void write(T x){
if(x < 0) putchar('-'), x = -x;
if(x >= 10) write(x / 10);
putchar('0' + x % 10);
}
char readchar(){
char c;
while(c = getchar(), c < 'A' || c > 'Z');
return c;
}
const int N = 100005;
int n, m;
int ecnt, adj[N], nxt[2*N], go[2*N];
int tot, pos[N], idx[N], fa[N], son[N], sze[N], top[N], dep[N], val[N];
int le[4*N], ri[4*N], data[4*N], lazy[4*N];
void add(int u, int v){
go[++ecnt] = v;
nxt[ecnt] = adj[u];
adj[u] = ecnt;
}
void pushup(int k){
data[k] = data[k << 1] + data[k << 1 | 1];
if(ri[k << 1] == le[k << 1 | 1]) data[k]--;
le[k] = le[k << 1], ri[k] = ri[k << 1 | 1];
}
void pushdown(int k){
if(lazy[k] == -1) return;
lazy[k << 1] = lazy[k << 1 | 1] = lazy[k];
le[k << 1] = le[k << 1 | 1] = lazy[k];
ri[k << 1] = ri[k << 1 | 1] = lazy[k];
data[k << 1] = data[k << 1 | 1] = 1;
lazy[k] = -1;
}
void build(int k, int l, int r){
lazy[k] = -1;
if(l == r) return (void)(data[k] = 1, le[k] = ri[k] = val[idx[l]]);
int mid = (l + r) >> 1;
build(k << 1, l, mid);
build(k << 1 | 1, mid + 1, r);
pushup(k);
}
void change(int k, int l, int r, int ql, int qr, int x){
if(ql <= l && qr >= r) return (void)(data[k] = 1, le[k] = ri[k] = lazy[k] = x);
pushdown(k);
int mid = (l + r) >> 1;
if(ql <= mid) change(k << 1, l, mid, ql, qr, x);
if(qr > mid) change(k << 1 | 1, mid + 1, r, ql, qr, x);
pushup(k);
}
int query(int k, int l, int r, int ql, int qr){
if(ql <= l && qr >= r) return data[k];
pushdown(k);
int mid = (l + r) >> 1;
if(qr <= mid) return query(k << 1, l, mid, ql, qr);
if(ql > mid) return query(k << 1 | 1, mid + 1, r, ql, qr);
return query(k << 1, l, mid, ql, qr) + query(k << 1 | 1, mid + 1, r, ql, qr) - (ri[k << 1] == le[k << 1 | 1]);
}
int getcol(int k, int l, int r, int p){
if(lazy[k] != -1) return lazy[k];
if(l == r) return le[k];
int mid = (l + r) >> 1;
if(pos[p] <= mid) return getcol(k << 1, l, mid, p);
else return getcol(k << 1 | 1, mid + 1, r, p);
}
void path_change(int u, int v, int x){
if(top[u] == top[v]){
if(pos[u] > pos[v]) swap(u, v);
change(1, 1, n, pos[u], pos[v], x);
return;
}
if(dep[top[u]] > dep[top[v]]) swap(u, v);
change(1, 1, n, pos[top[v]], pos[v], x);
path_change(u, fa[top[v]], x);
}
int path_query(int u, int v){
if(top[u] == top[v]){
if(pos[u] > pos[v]) swap(u, v);
return query(1, 1, n, pos[u], pos[v]);
}
if(dep[top[u]] > dep[top[v]]) swap(u, v);
int same = (getcol(1, 1, n, top[v]) == getcol(1, 1, n, fa[top[v]]));
return path_query(fa[top[v]], u) + query(1, 1, n, pos[top[v]], pos[v]) - same;
}
void init(){
static int que[N], qr;
que[qr = 1] = 1;
for(int ql = 1, u; ql <= qr; ql++){
u = que[ql], sze[u] = 1;
for(int e = adj[u], v; e; e = nxt[e])
if(v = go[e], v != fa[u])
fa[v] = u, dep[v] = dep[u] + 1, que[++qr] = v;
}
for(int ql = qr, u; ql; ql--){
u = que[ql];
sze[fa[u]] += sze[u];
if(sze[u] >= sze[son[fa[u]]]) son[fa[u]] = u;
}
for(int ql = 1, u; ql <= qr; ql++)
if(!top[u = que[ql]])
for(int v = u; v; v = son[v])
idx[++tot] = v, pos[v] = tot, top[v] = u;
build(1, 1, n);
}
int main(){
read(n), read(m);
for(int i = 1; i <= n; i++)
read(val[i]);
for(int i = 1, u, v; i < n; i++)
read(u), read(v), add(u, v), add(v, u);
init();
char op;
int a, b, c;
while(m--){
op = readchar(), read(a), read(b);
if(op == 'Q') write(path_query(a, b)), enter;
else read(c), path_change(a, b, c);
}
return 0;
}
BZOJ 2243 染色 | 树链剖分模板题进阶版的更多相关文章
- BZOJ 2243 染色 树链剖分
题意: 给出一棵树,每个顶点上有个颜色\(c_i\). 有两种操作: C a b c 将\(a \to b\)的路径所有顶点上的颜色变为c Q a b 查询\(a \to b\)的路径上的颜色段数,连 ...
- BZOJ - 2243 染色 (树链剖分+线段树+区间合并)
题目链接 线段树维护区间连续段个数即可.设lc为区间左端点颜色,rc为区间右端点颜色,则合并两区间的时候,如果左区间右端点和右区间左端点颜色相同,则连续段个数-1. 在树链上的区间合并可以定义一个结构 ...
- BZOJ 1036 树的统计Count 树链剖分模板题
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...
- bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题
[ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...
- hysbz 2243 染色(树链剖分)
题目链接:hysbz 2243 染色 题目大意:略. 解题思路:树链剖分+线段树的区间合并,可是区间合并比較简单,节点仅仅要记录左右端点的颜色就可以. #include <cstdio> ...
- 洛谷 P3384 树链剖分(模板题)
题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...
- 洛谷 P2146 [NOI2015]软件包管理器 (树链剖分模板题)
题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个 ...
- spoj - Grass Planting(树链剖分模板题)
Grass Planting 题意 给出一棵树,树有边权.每次给出节点 (u, v) ,有两种操作:1. 把 u 到 v 路径上所有边的权值加 1.2. 查询 u 到 v 的权值之和. 分析 如果这些 ...
- P3833 [SHOI2012]魔法树 (树链剖分模板题)
题目链接:https://www.luogu.org/problem/P3833 题目大意:有一颗含有n个节点的树,初始时每个节点的值为0,有以下两种操作: 1.Add u v d表示将点u和v之间的 ...
随机推荐
- python通讯录系统
---恢复内容开始--- 对于一般的通讯录系统,主要有两个参数:姓名和电话号码,所以可以利用python编程里面的字典来进行建立之间的所属关系, 可以利用以下代码简单实现: print('|--- 欢 ...
- 用UGUI制作可根据手指位置自动定位的隐形遥杆
之前写过遥杆怎么做,这里依然用的是之前的方法,就不介绍了. 之前玩过<蜡烛人>,发现手游版的<蜡烛人>的遥杆是看不见的,手指直接在屏幕左边滑动人物就可以移动,可能是为了增强沉浸 ...
- CSS基础范例
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- hostname命令详解
基础命令学习目录首页 原文链接:https://idc.wanyunshuju.com/cym/68.html Linux操作系统的hostname是一个kernel变量,可以通过hostname命令 ...
- uptime命令详解
基础命令学习目录首页 users个数和窗口数一致 原文链接:https://www.cnblogs.com/ultranms/p/9253217.html uptime 另外还有一个参数 -V(大写) ...
- 定时任务crone表达式demo
1. cron表达式格式: {秒数} {分钟} {小时} {日期} {月份} {星期} {年份(可为空)} 2. cron表达式各占位符解释: {秒数} ==> 允许值范围: 0~59 ,不允许 ...
- ModelAndView在页面上取值时value="{contextConfigLocation=<NONE>}"
后台: mv.getModel().put("initParam", 1); 页面: <input type="hidden" id="init ...
- ListView高效分页
使用控件自带的分页功能时,会先将所查询的数据全部加载出来,若数据量较大,则造成浏览器端等待时间过长. 然而在庞大的数据量,用户所需要的不过是那么几条,甚至只要其中的一条数据,所以,为了减少开销,每次只 ...
- java 框架 面试 2
1.谈谈你对Struts的理解. 答: 1. struts是一个按MVC模式设计的Web层框架,其实它就是一个大大的servlet,这个Servlet名为ActionServlet,或是ActionS ...
- linshi18
#include<iostream> using namespace std; int n,m,k; #define max 100 char mmap[max][max]; int mm ...