题目大意:给你一棵树,有两个操作:

  1. $C\;x:$给第$x$个节点打上标记
  2. $Q\;x:$询问第$x$个节点的祖先中最近的打过标记的点(自己也是自己的祖先)

题解:树剖,可以维护区间或,然后若一段区间为$0$则跳过,否则在线段树上二分

卡点:二分部分多大了一个$=$,然后$MLE$

C++ Code:

#include <cstdio>
#include <algorithm>
#include <ctime>
#include <cctype>
#include <cstdlib>
namespace __IO {
namespace R {
int x, ch;
inline int read() {
ch = getchar();
while (isspace(ch)) ch = getchar();
for (x = ch & 15, ch = getchar(); isdigit(ch); ch = getchar()) x= x * 10 + (ch & 15);
return x;
}
inline char readc() {
ch = getchar();
while (!isalpha(ch)) ch = getchar();
return static_cast<char> (ch);
}
}
}
using __IO::R::read;
using __IO::R::readc; #define maxn 100010
int head[maxn], cnt;
struct Edge {
int to, nxt;
} e[maxn << 1];
inline void add(int a, int b) {
e[++cnt] = (Edge) {b, head[a]}; head[a] = cnt;
e[++cnt] = (Edge) {a, head[b]}; head[b] = cnt;
} int n, m; int dfn[maxn], idx, fa[maxn], sz[maxn];
int son[maxn], top[maxn], dep[maxn], ret[maxn];
void dfs1(int u) {
sz[u] = 1;
for (int i = head[u]; i; i = e[i].nxt) {
int v = e[i].to;
if (v != fa[u]) {
dep[v] = dep[u] + 1;
fa[v] = u;
dfs1(v);
if (!son[u] || sz[v] > sz[son[u]]) son[u] = v;
sz[u] += sz[v];
}
}
}
void dfs2(int u) {
dfn[u] = ++idx, ret[idx] = u;
int v = son[u];
if (v) top[v] = top[u], dfs2(v);
for (int i = head[u]; i; i = e[i].nxt) {
int v = e[i].to;
if (v != son[u] && v != fa[u]) {
top[v] = v;
dfs2(v);
}
}
} namespace SgT {
bool V[maxn << 2];
int pos, L, R; void modify(int rt, int l, int r) {
V[rt] = true;
if (l == r) return ;
int mid = l + r >> 1;
if (pos <= mid) modify(rt << 1, l, mid);
else modify(rt << 1 | 1, mid + 1, r);
}
void modify(int __pos) {
pos = __pos;
modify(1, 1, n);
} bool res;
void query(int rt, int l, int r) {
if (L <= l && R >= r) return static_cast<void> (res |= V[rt]);
int mid = l + r >> 1;
if (L <= mid) query(rt << 1, l, mid);
if (res) return ;
if (R > mid) query(rt << 1 | 1, mid + 1, r);
}
bool query(int __L, int __R) {
res = false;
L = __L, R = __R;
query(1, 1, n);
return res;
} int ans;
void ask(int rt, int l, int r) {
if (!V[rt]) return ;
if (l == r) {
if (!ans) ans = l;
return ;
}
int mid = l + r >> 1;
if (R > mid) ask(rt << 1 | 1, mid + 1, r);
if (ans) return ;
if (L <= mid) ask(rt << 1, l, mid);
}
int ask(int __L, int __R) {
L = __L, R = __R;
ans = 0;
ask(1, 1, n);
return ret[ans];
}
} int query(int x) {
while (top[x] != 1) {
if (SgT::query(dfn[top[x]], dfn[x])) return SgT::ask(dfn[top[x]], dfn[x]);
x = fa[top[x]];
}
return SgT::ask(1, dfn[x]);
} int main() {
n = read(), m = read();
for (int i = 1, a, b; i < n; i++) {
a = read(), b = read();
add(a, b);
}
dfs1(1);
top[1] = 1;
dfs2(1); SgT::modify(1);
while (m --> 0) {
char op = readc();
int x = read();
if (op == 'C') {
SgT::modify(dfn[x]);
} else {
printf("%d\n", query(x));
}
}
return 0;
}

  

最新文章

  1. ubuntu:activate root
  2. KMP算法 hdu4686 Oulipo
  3. jquery改变元素的值的函数text(),html(),val()
  4. [kuangbin带你飞]专题七 线段树
  5. Quartz的misfire特性
  6. ember.js
  7. C#中判断字符串是否中文的方法
  8. Video.js网页视频播放插件
  9. 图像操作相关 With Quartz 2D
  10. 图的M着色问题
  11. mysql排序
  12. NHibernate与IbatisNet的简单比较
  13. pom大全
  14. Redis Lua脚本调试
  15. 洛谷 P1430 序列取数 解题报告
  16. angularjs中阻止事件冒泡,以及指令的注意点
  17. css 单行/多行文字垂直居中问题
  18. 八叉树(Octree)
  19. JS中eval处理JSON数据 为什么要加括号
  20. Objective-C语法之字符串NSString去掉前后空格或回车符(可以是NSCharacterSet类型的其它字符)

热门文章

  1. “地表最贵iPhone”到货,iPhone XS 系列手机等你来测!
  2. 了解Python控制流语句——break 语句
  3. JavaScript --经典问题
  4. jQuery的图片懒加载
  5. python3爬虫-快速入门-爬取图片和标题
  6. typescript 学习记录
  7. Stunnel客户端安装和配置
  8. nodejs笔记--Events篇(二)
  9. 条款03 尽可能使用const
  10. 欢迎来怼--第二十一次Scrum会议