ZOJ 3686 A Simple Tree Problem(线段树)
Description
Given a rooted tree, each node has a boolean (0 or 1) labeled on it. Initially, all the labels are 0.
We define this kind of operation: given a subtree, negate all its labels.
And we want to query the numbers of 1's of a subtree.
Input
Multiple test cases.
First line, two integer N and M, denoting the numbers of nodes and numbers of operations and queries.(1<=N<=100000, 1<=M<=10000)
Then a line with N-1 integers, denoting the parent of node 2..N. Root is node 1.
Then M lines, each line are in the format "o node" or "q node", denoting we want to operate or query on the subtree with root of a certain node.
Output
For each query, output an integer in a line.
Output a blank line after each test case.
题目大意:给一棵多叉树,初始值都为0,o x为翻转以x为根的子树,q x为查询以x为根的子树有多少个1
思路:这数据范围,暴力是不行的,怎么暴力都是不行的>_<。这题的要求是:修改一大片、查询一大片,比较容易想到的就是线段树(树状数组也可以,不过要翻转嘛……好像有难度……反正我不会>_<)。问题是这玩意儿怎么转换成线段树呢?要转化成线段树,就要把每个点的子孙们都放到一片连续的空间里。这时,若使用DFS,遍历的顺序刚刚好符合要求,于是我们就DFSo(∩_∩)o 。DFS途中就可以算出每个点的及其子孙覆盖的区域。然后变成线段树之后呢,随便搞搞就行了o(∩_∩)o
- #include <cstdio>
- #include <cstring>
- const int MAX = ;
- int flip[MAX*], sum[MAX*], cnt[MAX*];//tree
- int head[MAX], next[MAX], to[MAX], ecnt;
- int beg[MAX], size[MAX], dfs_clock;
- int y1, y2;
- void tle() {while() ;}
- void init() {
- ecnt = ;
- dfs_clock = ;
- memset(head, , sizeof(head));
- memset(flip, , sizeof(flip));
- memset(cnt, , sizeof(cnt));
- }
- void add_edge(int u, int v) {
- to[ecnt] = v; next[ecnt] = head[u]; head[u] = ecnt++;
- }
- void dfs(int x) {
- size[x] = ;
- beg[x] = ++dfs_clock;
- for(int p = head[x]; p; p = next[p]) {
- dfs(to[p]);
- size[x] += size[to[p]];
- }
- }
- void maintain(int x, int l, int r) {
- int lc = x * , rc = x * + ;
- if(l < r) {
- cnt[x] = cnt[rc] + cnt[lc];
- }
- }
- void pushdown(int x) {
- int lc = x * , rc = x * + ;
- if(flip[x]) {
- flip[x] = ;
- flip[lc] ^= ;
- cnt[lc] = sum[lc] - cnt[lc];
- flip[rc] ^= ;
- cnt[rc] = sum[rc] - cnt[rc];
- }
- }
- void update(int x, int l, int r) {
- int lc = x * , rc = x * + ;
- if(y1 <= l && r <= y2) {
- flip[x] ^= ;
- cnt[x] = sum[x] - cnt[x];
- }
- else {
- pushdown(x);
- int mid = (l + r) / ;
- if(y1 <= mid) update(lc, l, mid);
- if(mid < y2) update(rc, mid + , r);
- maintain(x, l, r);
- }
- }
- int ans;
- void query(int x, int l, int r) {
- int lc = x * , rc = x * + ;
- if(y1 <= l && r <= y2) ans += cnt[x];
- else {
- pushdown(x);
- int mid = (l + r) / ;
- if(y1 <= mid) query(lc, l, mid);
- if(mid < y2) query(rc, mid + , r);
- }
- }
- void build(int x, int l, int r) {
- int lc = x * , rc = x * + ;
- if(l == r) {
- sum[x] = ;
- }
- else {
- int mid = (l + r) / ;
- build(lc, l, mid);
- build(rc, mid + , r);
- sum[x] = sum[lc] + sum[rc];
- }
- }
- int main() {
- int n, m, x;
- char c[];
- while(scanf("%d%d", &n, &m) != EOF) {
- init();
- for(int i = ; i <= n; ++i) {
- scanf("%d", &x);
- add_edge(x, i);
- }
- dfs();
- build(, , n);
- while(m--) {
- scanf("%s%d", c, &x);
- y1 = beg[x]; y2 = beg[x] + size[x] - ;
- if(c[] == 'o') {
- update(, , n);
- }
- if(c[] == 'q') {
- ans = ;
- query(, , n);
- printf("%d\n", ans);
- }
- }
- puts("");
- }
- }
ZOJ 3686 A Simple Tree Problem(线段树)的更多相关文章
- ZOJ 3686 A Simple Tree Problem
A Simple Tree Problem Time Limit: 3 Seconds Memory Limit: 65536 KB Given a rooted tree, each no ...
- zoj 3686 A Simple Tree Problem (线段树)
Solution: 根据树的遍历道的时间给树的节点编号,记录下进入节点和退出节点的时间.这个时间区间覆盖了这个节点的所有子树,可以当做连续的区间利用线段树进行操作. /* 线段树 */ #pragma ...
- ZOJ-3686 A Simple Tree Problem 线段树
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3686 题意:给定一颗有根树,每个节点有0和1两种值.有两种操作: ...
- bzoj 3489 A simple rmq problem - 线段树
Description 因为是OJ上的题,就简单点好了.给出一个长度为n的序列,给出M个询问:在[l,r]之间找到一个在这个区间里只出现过一次的数,并且要求找的这个数尽可能大.如果找不到这样的数,则直 ...
- hdu 4973 A simple simulation problem. (线段树)
题目链接 题意: 给定n长的序列 m个操作 序列默认为 1, 2, 3···n 操作1:D [l,r] 把[l,r]区间增长 :( 1,2,3,4 进行 D [1,3]变成 1,1,2,2,3,3,4 ...
- BNU 28887——A Simple Tree Problem——————【将多子树转化成线段树+区间更新】
A Simple Tree Problem Time Limit: 3000ms Memory Limit: 65536KB This problem will be judged on ZJU. O ...
- 【BZOJ4999】This Problem Is Too Simple!(线段树)
[BZOJ4999]This Problem Is Too Simple!(线段树) 题面 BZOJ 题解 对于每个值,维护一棵线段树就好啦 动态开点,否则空间开不下 剩下的就是很简单的问题啦 当然了 ...
- xtu数据结构 I. A Simple Tree Problem
I. A Simple Tree Problem Time Limit: 3000ms Memory Limit: 65536KB 64-bit integer IO format: %lld ...
- hdu 5274 Dylans loves tree(LCA + 线段树)
Dylans loves tree Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Othe ...
随机推荐
- 常见web漏洞
常见的web漏洞——文件上传漏洞 一.文件上传漏洞概述 文件上传漏洞是指用户上传了一个可执行的脚本文件,并通过此脚本文件获得了执行服务器端命令的能力.这种攻击方式是最为直接和有效的,有时候几乎没 ...
- Linux中文件I/O函数
一.lseek函数 每个打开文件都有一个与其相关联的“当前文件偏移量”.它通常是一个非负整数,用以度量从文件开始处 计算的字节数.通常,读.写操作都从当前文件偏移量处开始,并使偏移量增加所读写的字节数 ...
- 竞赛题解 - NOIP2018 保卫王国
\(\mathcal{NOIP2018}\) 保卫王国 - 竞赛题解 按某一个炒鸡dalao名曰 taotao 的话说: \(\ \ \ \ \ \ \ \ \ "一道sb倍增题" ...
- MySQL 5.7增强半同步测试
we've know the machenism of semi-synchronous replication in my previous article,let's do som ...
- MySQL正则表达式的问题
原本以为 正则表达式里面的特殊\d匹配数字放到sql语句里面也是适用的,没想到一直不匹配.但是放到编程语言java或者js里面又匹配.看了一下原来sql对正则的支持没有那么全面.一定要用[0-9]代表 ...
- Python-调用系统指令小记
import subprocess def exec_command(cmd, log_path, **kwargs): with open(log_path, 'w') as f: p = subp ...
- Flask之蓝图的使用
蓝图,听起来就是一个很宏伟的东西 在Flask中的蓝图 blueprint 也是非常宏伟的 它的作用就是将 功能 与 主服务 分开怎么理解呢? 比如说,你有一个客户管理系统,最开始的时候,只有一个查看 ...
- html 的radio单选框如何实现互斥------radio只是input的type属性
先看看没有互斥的情况: <html> <body> 男性:<input type="radio" id="male" /> ...
- 最小化的测试套件minimal_test的使用
1:需要包含文件文#include <boost/test/minimal_test.hpp> 2:minimal_test内部实现了main(), 因此无需自己编写main()函数, 只 ...
- Random类与Random方法
class Test{ public static void main(String[] args){ int min=2; //定义随机数的最小值 int max=102; //定义随机数的最大值 ...