LCT强强!以前总是觉得LCT非常的难懂(当然现在也是的),但实际上它真的是很厉害的一种东西。它是一种动态的链剖分结构,其实就是对于剖分出来的重链使用LCT去进行维护。cut 与 link 两个操作让我们可以构造出希望存在的链(动态更新),而 split 操作则可以提取出任意一条从 \(u\) 到 \(v\) 的链使得这条链成为重链,也就是处于一棵 splay 当中,并用根节点来返回信息。可以说,大部分与链有关的问题都可以考虑使用LCT来求解。

  那么这道题乍一看1操作十分的棘手,但如果和LCT联想的话会发现实际上这就是一个 access 的操作,让一个节点到根的路径成为重链并断开原有的重儿子。这样一个条路径上的颜色个数,就是经过的虚边的条数。对于操作2,实际上是一个 LCT 的模板应用。我们维护颜色段的信息,用 \(LC[u]\) 和 \(RC[u]\) 分别代表以\(u\) 为根节点在 splay 上的子树的最右和最左两个节点的颜色。这样就可以维护了。修改颜色使用一个标记即可。

  而3操作我们可以发现:在 access 的时候,我们每修改一条虚边为实边,就会断开一条原本为实边的边为虚边。这样对于它们子树的贡献分别是 \(-1, +1\)。我们在线段树上维护一下加减值即可。不过要注意由于3操作中的 access 已经用于维护树的形态了,我们就不能再随意的去变动它。所以为了实现2和3操作,我们必须使用两棵LCT。以及在翻转子树的时候,\(LC[u]\) 和 \(RC[u]\) 都必须要翻转!

  这里我也有一个不是很理解的地方。标记一般来说有两种,一种打了标记表示自身及子树还未被修改,另一种则表示自身已经被修改,子树还未被修改。在以往我写的 LCT 中,翻转标记用第一种来维护完全没有问题。可以加入了覆盖的标记之后,只有第2种才是正确的。我也不是很懂为什么……如果有知道的,还请评论 \ 私信我一下好吗QAQ

  加之以前我一直以为LCT只能维护链信息,但老师说也是可以维护子树信息的,只要在添加虚边的时候更新一下即可,但这个必须要子树贡献满足可加性才行。

#include <bits/stdc++.h>
using namespace std;
#define maxn 1000000
int n, m, timer, size[maxn];
int id[maxn], dfn[maxn], dep[maxn]; int read()
{
int x = , k = ;
char c;
c = getchar();
while(c < '' || c > '') { if(c == '-') k = -; c = getchar(); }
while(c >= '' && c <= '') x = x * + c - '', c = getchar();
return x * k;
} struct edge
{
int cnp, to[maxn], last[maxn], head[maxn];
edge() { cnp = ; }
void add(int u, int v)
{
to[cnp] = v, last[cnp] = head[u], head[u] = cnp ++;
to[cnp] = u, last[cnp] = head[v], head[v] = cnp ++;
}
}E1; struct Segament_Tree
{
int mark[maxn], mx[maxn]; void Build(int p, int l, int r)
{
if(l == r) { mx[p] = dep[id[l]]; return; }
int mid = (l + r) >> ;
Build(p << , l, mid), Build(p << | , mid + , r);
mx[p] = max(mx[p << ], mx[p << | ]);
} void push_down(int u)
{
if(!mark[u]) return;
mark[u << ] += mark[u]; mark[u << | ] += mark[u];
mx[u << ] += mark[u], mx[u << | ] += mark[u];
mark[u] = ;
} void update(int p, int l, int r, int L, int R, int x)
{
if(L <= l && R >= r)
{
mark[p] += x; mx[p] += x;
return;
}
if(L > r || R < l) return;
int mid = (l + r) >> ;
push_down(p);
update(p << , l, mid, L, R, x); update(p << | , mid + , r, L, R, x);
mx[p] = max(mx[p << ], mx[p << | ]);
} int query(int p, int l, int r, int L, int R)
{
if(L <= l && R >= r) return mx[p];
if(L > r || R < l) return ;
push_down(p); int mid = (l + r) >> ;
return max(query(p << , l, mid, L, R), query(p << | , mid + , r, L, R));
}
}ST; struct Link_Cut_Tree
{
int fa[maxn], rev[maxn], ch[maxn][];
int sum[maxn], LC[maxn], RC[maxn], col[maxn];
int tot, flag, mark[maxn], top[maxn];
bool is_root(int u) { return (ch[fa[u]][] != u) && ( ch[fa[u]][] != u ); }
void Modify(int u, int x) { if(!u) return; mark[u] = x; LC[u] = RC[u] = col[u] = x, sum[u] = ; }
void Rev(int u) { if(!u) return; rev[u] ^= ; swap(ch[u][], ch[u][]); swap(LC[u], RC[u]); } void push_down(int u)
{
if(!is_root(u)) push_down(fa[u]);
if(rev[u]) Rev(ch[u][]), Rev(ch[u][]), rev[u] = ;
if(mark[u]) Modify(ch[u][], mark[u]), Modify(ch[u][], mark[u]), mark[u] = ;
} void update(int u)
{
int lc = ch[u][], rc = ch[u][];
if(flag)
{
sum[u] = sum[lc] + sum[rc] + ((col[u] != RC[lc]) && (col[u] != LC[rc]));
sum[u] -= ((col[u] == RC[lc]) && (col[u] == LC[rc]));
LC[u] = LC[lc], RC[u] = RC[rc];
if(!lc) LC[u] = col[u]; if(!rc) RC[u] = col[u];
}
else top[u] = top[lc] ? top[lc] : u;
} void rotate(int u)
{
int f = fa[u], gf = fa[f];
int k = ch[f][] == u;
fa[u] = gf; if(!is_root(f)) ch[gf][ch[gf][] == f] = u;
ch[f][k] = ch[u][k ^ ], fa[ch[u][k ^ ]] = f;
fa[f] = u, ch[u][k ^ ] = f;
update(f), update(u);
} void Splay(int u)
{
push_down(u);
while(!is_root(u))
{
int f = fa[u], gf = fa[f];
if(!is_root(f)) (ch[gf][] == f) ^ (ch[f][] == u) ? rotate(u) : rotate(f);
rotate(u);
}
update(u);
} void Access(int u)
{
for(int i = u, last = ; i; last = i, i = fa[i])
{
Splay(i);
if(!flag)
{
int x = top[last], y = top[ch[i][]];
if(x) ST.update(, , n, dfn[x], dfn[x] + size[x] - , -);
if(y) ST.update(, , n, dfn[y], dfn[y] + size[y] - , );
}
ch[i][] = last; update(i);
}
}
void Make_root(int u) { Access(u); Splay(u); Rev(u); }
void Split(int u, int v) { Make_root(u); Access(v); Splay(v); } void Change1(int u) { ++ tot; Access(u); }
void Change2(int u) { Split(, u); Modify(u, ++ tot); }
void Link(int u, int v) { Make_root(u); fa[u] = v; } int Query(int u, int v) { Split(u, v); return sum[v]; }
}LCT1, LCT2; void dfs(int u, int fa)
{
LCT1.fa[u] = fa;
dfn[u] = ++ timer; size[u] = ; dep[u] = dep[fa] + ;
id[timer] = u; LCT1.top[u] = u;
for(int i = E1.head[u]; i; i = E1.last[i])
{
int v = E1.to[i];
if(v == fa) continue;
dfs(v, u); size[u] += size[v];
}
} int main()
{
n = read(), m = read(); LCT2.flag = ;
for(int i = ; i <= n; i ++)
LCT2.col[i] = LCT2.RC[i] = LCT2.LC[i] = ++ LCT2.tot, LCT2.sum[i] = ;
for(int i = ; i < n; i ++)
{
int x = read(), y = read();
E1.add(x, y); LCT2.Link(x, y);
}
dfs(, );
ST.Build(, , n);
for(int i = ; i <= m; i ++)
{
int opt = read();
if(opt == )
{
int x = read();
LCT1.Change1(x); LCT2.Change2(x);
}
else if(opt == )
{
int x = read(), y = read();
printf("%d\n", LCT2.Query(x, y));
}
else if(opt == )
{
int x = read();
printf("%d\n", ST.query(, , n, dfn[x], dfn[x] + size[x] - ));
}
}
return ;
}

【题解】SDOI2017树点涂色的更多相关文章

  1. 【LG3703】[SDOI2017]树点涂色

    [LG3703][SDOI2017]树点涂色 题面 洛谷 题解 更博辣,更博辣!!! 猪年的第一篇博客 一次只能染根到\(x\),且染的颜色未出现过 这句话是我们解题的关键. 设\(x\)到根的颜色数 ...

  2. 【BZOJ4817】[Sdoi2017]树点涂色 LCT+线段树

    [BZOJ4817][Sdoi2017]树点涂色 Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路径的权值是:这条路 ...

  3. [Bzoj4817] [Sdoi2017]树点涂色 (LCT神题)

    4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 629  Solved: 371[Submit][Status ...

  4. [Sdoi2017]树点涂色 [lct 线段树]

    [Sdoi2017]树点涂色 题意:一棵有根树,支持x到根染成新颜色,求x到y颜色数,求x子树里点到根颜色数最大值 考场发现这个信息是可减的,但是没想到lct 特意设计成lct的形式! 如何求颜色数? ...

  5. P3703 [SDOI2017]树点涂色

    P3703 [SDOI2017]树点涂色 链接 分析: 首先对于询问,感觉是线段树维护dfs序,每个点记录到根的颜色个数.第二问差分,第三问区间取max. 那么考虑修改,每次将一个点的颜色变成和父节点 ...

  6. [BZOJ4817][SDOI2017]树点涂色(LCT+DFS序线段树)

    4817: [Sdoi2017]树点涂色 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 692  Solved: 408[Submit][Status ...

  7. BZOJ 4817 [SDOI2017]树点涂色 (LCT+线段树维护dfs序)

    题目大意:略 涂色方式明显符合$LCT$里$access$操作的性质,相同颜色的节点在一条深度递增的链上 用$LCT$维护一个树上集合就好 因为它维护了树上集合,所以它别的啥都干不了了 发现树是静态的 ...

  8. BZOJ 4817: [Sdoi2017]树点涂色(LCT+树剖+线段树)

    题目描述 Bob有一棵 nn 个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. Bob ...

  9. P3703 [SDOI2017]树点涂色 LCT维护颜色+线段树维护dfs序+倍增LCA

    \(\color{#0066ff}{ 题目描述 }\) Bob有一棵\(n\)个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点 ...

随机推荐

  1. LeetCode: 60. Permutation Sequence(Medium)

    1. 原题链接 https://leetcode.com/problems/permutation-sequence/description/ 2. 题目要求 给出整数 n和 k ,k代表从1到n的整 ...

  2. Android stdio build.gradle buildscript 里面的repositories 和allprojects里面 repositories 的区别

    第一段 buildscript 里面的 repositories 表示只有编译工具才会用这个仓库. 比如 buildscript 里面的 dependencies classpath 'com.and ...

  3. dsp6657的串口学习

    1. 打算用dsp6657学习下,先用串口实验吧.找一下芯片支持库Chip support libraries,路径D:\ti\pdk_C6657_1_1_1_4\packages\ti\csl,新建 ...

  4. php安全性问题

    目录 常见攻击类型 1.sql注入: 2.xss攻击 3.csrf攻击: php安全三板斧:过滤输入.验证数据,以及转义输出. 1.数据过滤: 2.验证数据: 3.转义输出: laravel 中如何避 ...

  5. libevent学习七(bufferevent)

    1. 每个bufferevent 都拥有类型为struct evbuffer的input buffer和out buffer,分别供数据读取和数据写入使用. 2.读取和写入数据是通过编写和设置对应的回 ...

  6. 对JSON的理解

    JSON语法: JSON是一种结构化数据,它是一种数据格式 JSON可以概括为三种类型:简单值.对象.数组 注意:JSON不支持变量.函数和对象实例 一.JSON简单值 包括字符串.数值.布尔值.和n ...

  7. 「日常训练」More Cowbell(Codeforces Round #334 Div.2 B)

    题意与分析(CodeForces 604B) 题意是这样的:\(n\)个数字,\(k\)个盒子,把\(n\)个数放入\(k\)个盒子中,每个盒子最多只能放两个数字,问盒子容量的最小值是多少(水题) 不 ...

  8. 「国庆训练」Bomb(HDU-5934)

    题意 给定\(n\)个炸弹,每个炸弹的坐标与代价与影响范围给定,炸弹会引爆影响范围内其他所有炸弹.求引爆所有炸弹的最小代价. 分析 先做\(n^2\)的循环,然后建图,对\(i\)能引爆\(j\)建边 ...

  9. js 加密 crypto-js des加密

    js 加密 crypto-js    https://www.npmjs.com/package/crypto-js   DES  举例:   js 引入:   <script src=&quo ...

  10. Python全栈 Web(边框、盒模型、背景)

    原文地址 https://yq.aliyun.com/articles/634926 ......................................................... ...