1.别忘了初始化,初始时各个节点颜色互不相同,也就是ans = d(到根的距离)
2.rot时分清谁和谁, 我脑残写了 fa[x] = y , fa[y] = x
#include<iostream>
#include<cstdio>
using namespace std;
const int N = 210000;
inline int read()
{
register int x = 0; register char c = getchar();
while(c < '0' || c > '9') c = getchar(); while(c >= '0' && c <= '9') x = (x << 3) + (x << 1) + c - '0' , c = getchar();
return x;
}
int n , m , dfn , cnt;
int head[N] , st[N] , ed[N] , d[N] , fdfn[N] , f[N][20] , siz[N] , fa[N] , tr[N][2] , rev[N] , maxn[N<<2] , tag[N<<2];
struct edge{int v , nex; } e[N<<1];
inline void add(int u , int v) { e[++cnt].v = v; e[cnt].nex = head[u]; head[u] = cnt; return ; }
#define D_bag puts("this is OK!!");
// Segment_tree->begin()
void Down(int k)
{
if(tag[k])
{
maxn[k<<1] += tag[k]; maxn[k<<1|1] += tag[k];
tag[k<<1] += tag[k]; tag[k<<1|1] += tag[k]; tag[k] = 0; // tag -->clear
}
return ;
}
void build(int k , int l , int r) // 建树初始化
{
if(l == r)
{
maxn[k] = d[fdfn[l]];
return ;
}
int mid = (l + r) >> 1;
build(k << 1 , l , mid);
build(k << 1 | 1 , mid + 1 , r);
maxn[k] = max(maxn[k<<1] , maxn[k<<1|1]);
return ;
}
void Change(int k , int l , int r , int x , int y , int val)
{
if(x <= l && r <= y) return (void)(maxn[k] += val , tag[k] += val);
int mid = (l + r) >> 1; Down(k);
if(x <= mid) Change(k << 1 , l , mid , x , y , val);
if(y > mid) Change(k << 1 | 1 , mid+1 , r , x , y , val);
maxn[k] = max(maxn[k<<1] , maxn[k<<1|1]); return ;
}
int Ask(int k , int l , int r , int x , int y)
{
if(x <= l && r <= y) return maxn[k];
Down(k); int mid = (l + r) >> 1 , ans = 0;
if(x <= mid) ans = max(ans , Ask(k<<1 , l , mid , x , y));
if(y > mid) ans = max(ans , Ask(k<<1|1 , mid+1 , r , x , y));
return ans;
}
// Segment_tree->end()
inline int witch(int x) { return x == tr[fa[x]][1]; }
inline int nrt(int x) { return (fa[x] && (tr[fa[x]][0] == x || tr[fa[x]][1] == x)); }
void rot(int x)
{
int y = fa[x] , z = fa[y] , k = witch(x) , w = tr[x][!k];
if(nrt(y)) tr[z][witch(y)] = x; fa[x] = z; // !!!!
tr[y][k] = w; if(w) fa[w] = y;
tr[x][!k] = y; fa[y] = x;
}
void Splay(int x)
{
while(nrt(x))
{
if(nrt(fa[x])) rot(witch(x) == witch(fa[x]) ? fa[x] : x);
rot(x);
}
return ;
}
int find_son(int x)
{
while(tr[x][0]) x = tr[x][0]; return x;
}
void assess(int x)
{
int t = 0;
for(t = 0; x ; t = x , x = fa[x])
{
Splay(x);
if(t)
{
int son = find_son(t);
Change(1 , 1 , n , st[son] , ed[son] , -1);
}
if(tr[x][1])
{
int son = find_son(tr[x][1]);
Change(1 , 1 , n , st[son] , ed[son] , 1);
}
tr[x][1] = t;
}
return ;
}
int lca(int x , int y)
{
if(x == y) return x;
if(d[x] < d[y]) swap(x , y);
for(int i = 18 ; i >= 0 ; --i) if(d[f[x][i]] >= d[y]) x = f[x][i];
if(x == y) return x;
for(int i = 18 ; i >= 0 ; --i) if(f[x][i] != f[y][i]) x = f[x][i] , y = f[y][i];
return f[x][0];
}
void dfs(int x)
{
st[x] = ++dfn; fdfn[dfn] = x;
for(int i = 1 ; i <= 18 ; ++i) f[x][i] = f[f[x][i-1]][i-1];
for(int i = head[x] , v; i ; i = e[i].nex)
{
v = e[i].v;
if(v == f[x][0]) continue;
f[v][0] = x; d[v] = d[x] + 1; fa[v] = x; dfs(v);
}
ed[x] = dfn;
return ;
}
int main()
{
n = read(); m = read();
for(int i = 1 , a , b; i <= n - 1 ; ++i)
{
a = read(); b = read();
add(a , b); add(b , a);
}
d[1] = 1; dfs(1);
build(1 , 1 , n);
for(int i = 1 , op , x ; i <= m ; ++i)
{
op = read(); x = read();
if(op == 1) assess(x);
else
if(op == 2)
{
int y = read() , p = lca(x , y);
int ans = Ask(1 , 1 , n , st[x] , st[x]) + Ask(1 , 1 , n , st[y] , st[y]) - 2 * Ask(1 , 1 , n , st[p] , st[p]) + 1;
printf("%d\n" , ans);
}
else
if(op == 3)
printf("%d\n" , Ask(1 , 1 , n , st[x] , ed[x]));
}
return 0;
}
/*
5 6
1 2
2 3
3 4
3 5
2 4 5
3 3
1 4
2 4 5
1 5
2 4 5
*/
- BZOJ 4817 [SDOI2017]树点涂色 (LCT+线段树维护dfs序)
题目大意:略 涂色方式明显符合$LCT$里$access$操作的性质,相同颜色的节点在一条深度递增的链上 用$LCT$维护一个树上集合就好 因为它维护了树上集合,所以它别的啥都干不了了 发现树是静态的 ...
- bzoj 4817: [Sdoi2017]树点涂色
Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. ...
- 【刷题】BZOJ 4817 [Sdoi2017]树点涂色
Description Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路 径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. ...
- BZOJ 4817: [Sdoi2017]树点涂色(LCT+树剖+线段树)
题目描述 Bob有一棵 nn 个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. Bob ...
- BZOJ 4817: [Sdoi2017]树点涂色 LCT+Access的性质+DFS序+线段树
Code: #include<bits/stdc++.h> #define maxn 200003 #define inf -1000000 using namespace std; vo ...
- BZOJ.4817.[SDOI2017]树点涂色(LCT DFS序 线段树)
题目链接 操作\(1.2\)裸树剖,但是操作\(3\)每个点的答案\(val\)很不好维护.. 如果我们把同种颜色的点划分到同一连通块中,那么向根染色的过程就是Access()! 最初所有点间都是虚边 ...
- bzoj 4817: [Sdoi2017]树点涂色 LCT+树链剖分+线段树
题目: Bob有一棵n个点的有根树,其中1号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同. 定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜色. Bob可能会进 ...
- bzoj 4817: [Sdoi2017]树点涂色【树链剖分+LCT】
非常妙的一道题. 首先对于操作一"把点x到根节点的路径上所有的点染上一种没有用过的新颜色",长得是不是有点像LCT中的access操作?进而发现,如果把同一颜色的点连起来作为LCT ...
- BZOJ 4817 [Sdoi2017]树点涂色 ——LCT 线段树
同BZOJ3779. SDOI出原题,还是弱化版的. 吃枣药丸 #include <map> #include <cmath> #include <queue> # ...
随机推荐
- C#中StreamReader类读取文件使用示例
C#中StreamReader类读取文件使用示例 1.需要导入的命名空间是:System.IO; 2.操作的是字符,所以打开的是文本文件. 常用属性: CurrentEncoding:对象正在使用 ...
- springCloud进阶(微服务架构&Eureka)
springCloud进阶(微服务架构&Eureka) 1. 微服务集群 1.1 为什么要集群 为了提供并发量,有时同一个服务提供者可以部署多个(商品服务).这个客户端在调用时要根据一定的负责 ...
- System.Text.Json 自定义Converter实现时间转换
Newtonsoft.Json与System.Text.Json区别 在 Newtonsoft.Json中可以使用例如 .AddJsonOptions(options => { options. ...
- JSP开发机票预定系统 源码
开发环境: Windows操作系统开发工具:MyEclipse+Jdk+Tomcat6+Mysql数据库 运行效果图 源码及原文链接:https://javadao.xyz/forum.php?mod ...
- C语言 运算符
C语言 运算符 运算符优先级别 优先级 运算符 名称或含义 使用形式 结合方向 说明 1 [] 数组下标 数组名[常量表达式] 左到右 -- () 圆括号 (表达式)/函数名(形参表) -- . 成员 ...
- ClrFromCSharp_2_2_生成部署打包应用程序
1,在\reps\CSharpFromCSarp\CSharpFromCSarp_2_2,建立新解决方案和解决项目 并且输入以下代码 namespace ClrFromCSharp_2_2 { cla ...
- JavaScript自学笔记(3)--- 用JS来实现网页浮窗
最近做个小项目,给网页加个浮窗,考验了基础的css,js技术,还是蛮有意思的,代码如下(部分代码来源于引用,见底部) <!DOCTYPE html> <html> <he ...
- beego的请求数据处理
我们经常需要获取用户传递的数据,包括 Get.POST 等方式的请求,beego 里面会自动解析这些数据,你可以通过如下方式获取数据: GetString(key string) string Get ...
- ajax-属性、原理、实现html5进度条上传文件
一.远古ajax实现方式如下: 1.前端请求后台,后台设置返回 http状态码204 2.运用img图片(或css/javascript等)的加载机制,请求后台 3.post提交数据,运用iframe ...
- [P5748] 集合划分计数 - 生成函数,NTT
求 \(10^5\) 以内的所有贝尔数:将 \(n\) 个有标号的球划分为若干非空集合的方案数 Solution 非空集合的指数生成函数为 \(F(x)=e^x-1\) 枚举一共用多少个集合,答案就是 ...
|