【cf343】D. Water Tree(dfs序+线段树)

时间:2021-09-26 13:43:57

传送门

题意:

给出一个以\(1\)为根的有根树,起始每个结点都为\(0\),现在有三种操作:

  • 1.将\(v\)及\(v\)的子树都置为\(1\);
  • 2.将\(v\)及其所有的祖先都置为\(0\);
  • 3.询问\(v\)目前处于何种状态。

对于每次询问给出回答。

思路:

  • 对于\(1\)操作,子树修改,显然\(dfs\)序+线段树即可解决。
  • \(2\)操作除开树链剖分这种,很难去高效维护。但是注意到如果一个结点\(v\)为\(0\),那么其所有祖先都为\(0\)。那么对于\(2\)操作,直接单点修改,在操作\(1\)子树修改之前先看看子树中是否含有\(0\),若含有,则把当前根节点的父亲置为\(0\)即可。
  • 对于\(3\)操作同\(2\)操作查询子树最小值即可。

这个题本来想\(dsu\ on\ tree\)来搞,但是感觉离线确实不是很好做,所以写了个在线的方法。

感觉这个方法挺巧妙的,观察到性质之后,并不直接去维护链的信息,有点类似于线段树的懒惰标记,直接将\(2\)操作转换为单点修改了。

/*
* Author: heyuhhh
* Created Time: 2019/11/15 11:03:07
*/
#include <bits/stdc++.h>
#define MP make_pair
#define fi first
#define se second
#define sz(x) (int)(x).size()
#define all(x) (x).begin(), (x).end()
#define INF 0x3f3f3f3f
#define Local
#ifdef Local
#define dbg(args...) do { cout << #args << " -> "; err(args); } while (0)
void err() { std::cout << '\n'; }
template<typename T, typename...Args>
void err(T a, Args...args) { std::cout << a << ' '; err(args...); }
#else
#define dbg(...)
#endif
void pt() {std::cout << '\n'; }
template<typename T, typename...Args>
void pt(T a, Args...args) {std::cout << a << ' '; pt(args...); }
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
//head
const int N = 500005; int n;
vector <int> g[N];
int f[N], in[N], out[N], dfn; void dfs(int u, int fa) {
f[u] = fa;
in[u] = ++dfn;
for(auto v : g[u]) if(v != fa) {
dfs(v, u);
}
out[u] = dfn;
} int minv[N << 2], lz[N << 2]; void push_down(int o, int l, int r) {
if(lz[o] != -1) {
lz[o << 1] = lz[o << 1|1] = lz[o];
minv[o << 1] = minv[o << 1|1] = lz[o];
lz[o] = -1;
}
} void build(int o, int l, int r) {
minv[o] = 0; lz[o] = -1;
if(l == r) return;
int mid = (l + r) >> 1;
build(o << 1, l, mid);
build(o << 1|1, mid + 1, r);
} void upd(int o, int l, int r, int L, int R, int v) {
if(L <= l && r <= R) {
minv[o] = lz[o] = v;
return;
}
push_down(o, l, r);
int mid = (l + r) >> 1;
if(L <= mid) upd(o << 1, l, mid, L, R, v);
if(R > mid) upd(o << 1|1, mid + 1, r, L, R, v);
minv[o] = min(minv[o << 1], minv[o << 1|1]);
} int query(int o, int l, int r, int L, int R) {
if(L <= l && r <= R) return minv[o];
push_down(o, l, r);
int res = 2;
int mid = (l + r) >> 1;
if(L <= mid) res = query(o << 1, l, mid, L, R);
if(R > mid) res = min(res, query(o << 1|1, mid + 1, r, L, R));
return res;
} void run(){
for(int i = 1; i < n; i++) {
int u, v; cin >> u >> v;
g[u].push_back(v);
g[v].push_back(u);
}
dfs(1, 0);
build(1, 1, n);
int q; cin >> q;
while(q--) {
int op, v; cin >> op >> v;
if(op == 1) {
int Min = query(1, 1, n, in[v], out[v]);
if(Min == 0 && v > 1) {
upd(1, 1, n, in[f[v]], in[f[v]], 0);
}
upd(1, 1, n, in[v], out[v], 1);
} else if(op == 2) {
upd(1, 1, n, in[v], in[v], 0);
} else {
int Min = query(1, 1, n, in[v], out[v]);
if(Min == 0) cout << 0 << '\n';
else cout << 1 << '\n';
}
}
} int main() {
ios::sync_with_stdio(false);
cin.tie(0); cout.tie(0);
cout << fixed << setprecision(20);
while(cin >> n) run();
return 0;
}

【cf343】D. Water Tree(dfs序+线段树)的更多相关文章

  1. POJ&period;3321 Apple Tree &lpar; DFS序 线段树 单点更新 区间求和&rpar;

    POJ.3321 Apple Tree ( DFS序 线段树 单点更新 区间求和) 题意分析 卡卡屋前有一株苹果树,每年秋天,树上长了许多苹果.卡卡很喜欢苹果.树上有N个节点,卡卡给他们编号1到N,根 ...

  2. POJ3321 - Apple Tree DFS序 &plus; 线段树或树状数组

    Apple Tree:http://poj.org/problem?id=3321 题意: 告诉你一棵树,每棵树开始每个点上都有一个苹果,有两种操作,一种是计算以x为根的树上有几个苹果,一种是转换x这 ...

  3. Codeforces Round &num;225 &lpar;Div&period; 2&rpar; E&period; Propagating tree dfs序&plus;-线段树

    题目链接:点击传送 E. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes inpu ...

  4. poj 3321 Apple Tree dfs序&plus;线段树

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K       Description There is an apple tree outsid ...

  5. codechef T6 Pishty and tree dfs序&plus;线段树

    PSHTTR: Pishty 和城堡题目描述 Pishty 是生活在胡斯特市的一个小男孩.胡斯特是胡克兰境内的一个古城,以其中世纪风格 的古堡和非常聪明的熊闻名全国. 胡斯特的镇城之宝是就是这么一座古 ...

  6. codeforces 620E&period; New Year Tree dfs序&plus;线段树&plus;bitset

    题目链接 给一棵树, 每个节点有颜色, 两种操作, 一种是将一个节点的子树全都染色成c, 一种是查询一个节点的子树有多少个不同的颜色, c<=60. 每个节点一个bitset维护就可以. #in ...

  7. CodeForces 620E&colon;New Year Tree&lpar;dfs序&plus;线段树&rpar;

    E. New Year Treetime limit per test3 secondsmemory limit per test256 megabytesinputstandard inputout ...

  8. Codeforces 343D Water Tree(DFS序 &plus; 线段树)

    题目大概说给一棵树,进行以下3个操作:把某结点为根的子树中各个结点值设为1.把某结点以及其各个祖先值设为0.询问某结点的值. 对于第一个操作就是经典的DFS序+线段树了.而对于第二个操作,考虑再维护一 ...

  9. Educational Codeforces Round 6 E dfs序&plus;线段树

    题意:给出一颗有根树的构造和一开始每个点的颜色 有两种操作 1 : 给定点的子树群体涂色 2 : 求给定点的子树中有多少种颜色 比较容易想到dfs序+线段树去做 dfs序是很久以前看的bilibili ...

随机推荐

  1. 利用Shodan和Censys进行信息侦查

    在渗透测试的初始阶段,Shodan.Censys等在线资源可以作为一个起点来识别目标机构的技术痕迹.本文中就以二者提供的Python API为例,举例介绍如何使用它们进行渗透测试初期的信息侦查. Sh ...

  2. android&period;net&period;wifi的简单使用方法

    获取Wifi的控制类WifiManager.  WifiManager  wm=(WifiManager)getSystemService(Context.WIFI_SERVICE); 接下来可以对w ...

  3. java笔记--用ThreadLocal管理线程,Callable&lt&semi;V&gt&semi;接口实现有返回值的线程

    用ThreadLocal管理线程,Callable<V>接口实现有返回值的线程 ThreadLocal在我的笔记"关于线程同步"的第5种方式里面有介绍,这里就不多说了. ...

  4. linux概念之&sol;proc与&sol;sys

    http://blog.chinaunix.net/uid-1835494-id-3070465.html  proc/x:1/sched http://bbs.chinaunix.net/threa ...

  5. 防止双击选中html中文字

    在开发过程中很常用的会给<span></span>等内联元素增加一个onlick事件,但是经常发生的一件事情就是点击的时候,选中了span中的字体:倒是不影响主逻辑,但是很难受 ...

  6. JZ2440开发笔记(8)——FCLK、HCLK和PCLK

    S3C2440中有三种时钟,分别是FCLK,HCLK和PCLK.这三种时钟的功能各不相同,其中FCLK主要是为ARM920T的内核提供工作频率,如图: HCLK主要是为S3C2440 AHB总线(Ad ...

  7. &lbrack;Java&rsqb; List &sol; ArrayList - 源代码学习笔记

    在阅读 List / ArrayList 源代码过程中,做了下面的笔记. LinkedList 的笔记较多,放到了另一篇博文 LinkedList / Queue- 源代码学习笔记 List List ...

  8. 个性化品牌开始繁荣?为设计师和代工厂牵线的平台Maker's Row获得100万美元融资 &vert; 36氪

    个性化品牌开始繁荣?为设计师和代工厂牵线的平台Maker's Row获得100万美元融资 | 36氪 个性化品牌开始繁荣?为设计师和代工厂牵线的平台Maker's Row获得100万美元融资

  9. mysql开启慢查询日志以及查看(转载自网络)

    转载自http://database.51cto.com/art/201309/410314_1.htm

  10. 007-declare 声明变量的类型

    declare [+/-] [选项] 变量名 - 给变量设定类型 + 取消变量的类型 -a 将变量声明为数组型 -i 将变量声明为整形 -x 将变量声明成环境变量 -r 将变量声明为只读变量 -p 显 ...