2014 Super Training #9 F A Simple Tree Problem --DFS+线段树
原题: ZOJ 3686 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3686
这题本来是一个比较水的线段树,结果一个mark坑了我好几个小时。。哎。太弱。
先DFS这棵树,树形结构转换为线性结构,每个节点有一个第一次遍历的时间和最后一次遍历的时间,之间的时间戳都为子树的时间戳,用线段树更新这段区间即可实现更新子树的效果,用到懒操作节省时间。
坑我的地方: update时,不能写成:tree[rt].mark = 1, 而要写成 tree[rt].mark ^= 1; 因为如果持续update偶数次相当于什么都没做,但是每次update确实是更新了的啊,每次pushdown都会将tree[rt].mark变为0的啊,也就是说tree[rt].mark是不能保持的,所以我搞不懂的是为什么要异或1,而不能直接令等于1,有待向大神请教,如果有看到并知道的仁兄可以指教一下我。
代码:
- #include <iostream>
- #include <cstdio>
- #include <cstring>
- #include <cmath>
- #include <algorithm>
- #include <vector>
- using namespace std;
- #define N 100007
- struct node
- {
- int l,r;
- }p[N];
- struct Tree
- {
- int sum,mark;
- }tree[*N];
- int Time;
- vector<int> G[N];
- void pushup(int rt)
- {
- tree[rt].sum = tree[*rt].sum + tree[*rt+].sum;
- }
- void build(int l,int r,int rt)
- {
- tree[rt].sum = tree[rt].mark = ;
- if(l == r)
- return;
- int mid = (l+r)/;
- build(l,mid,*rt);
- build(mid+,r,*rt+);
- pushup(rt);
- }
- void pushdown(int l,int r,int rt)
- {
- if(!tree[rt].mark)
- return;
- int mid = (l+r)/;
- tree[*rt].mark ^= ; //not "tree[2*rt].mark = tree[rt].mark"
- tree[*rt+].mark ^= ;
- tree[*rt].sum = (mid-l+)-tree[*rt].sum;
- tree[*rt+].sum = (r-mid)-tree[*rt+].sum;
- tree[rt].mark = ;
- }
- void update(int l,int r,int aa,int bb,int rt)
- {
- if(aa<=l && bb>=r)
- {
- tree[rt].mark ^= ; //not "tree[rt].mark = 1", 因为偶数次操作相互抵消
- tree[rt].sum = r-l+-tree[rt].sum;
- return;
- }
- pushdown(l,r,rt);
- int mid = (l+r)/;
- if(aa <= mid)
- update(l,mid,aa,bb,*rt);
- if(bb > mid)
- update(mid+,r,aa,bb,*rt+);
- pushup(rt);
- }
- void dfs(int u)
- {
- p[u].l = Time++;
- for(int i=;i<G[u].size();i++)
- dfs(G[u][i]);
- p[u].r = Time++;
- }
- int query(int l,int r,int aa,int bb,int rt)
- {
- if(aa>r || bb<l)
- return ;
- if(aa<=l && bb>=r)
- return tree[rt].sum;
- pushdown(l,r,rt);
- int mid = (l+r)/;
- return query(l,mid,aa,bb,*rt)+query(mid+,r,aa,bb,*rt+);
- }
- int main()
- {
- int n,m,i,j,x;
- char ss[];
- while(scanf("%d%d",&n,&m)!=EOF)
- {
- for(i=;i<=n;i++)
- G[i].clear();
- for(i=;i<=n;i++)
- {
- scanf("%d",&x);
- G[x].push_back(i);
- }
- Time = ;
- dfs();
- build(,*n,);
- while(m--)
- {
- scanf("%s%d",ss,&x);
- if(ss[] == 'o')
- update(,*n,p[x].l,p[x].r,);
- else
- printf("%d\n",query(,*n,p[x].l,p[x].r,)/);
- }
- puts("");
- }
- return ;
- }
2014 Super Training #9 F A Simple Tree Problem --DFS+线段树的更多相关文章
- 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 ...
- zoj 3686 A Simple Tree Problem (线段树)
Solution: 根据树的遍历道的时间给树的节点编号,记录下进入节点和退出节点的时间.这个时间区间覆盖了这个节点的所有子树,可以当做连续的区间利用线段树进行操作. /* 线段树 */ #pragma ...
- 2014 Super Training #7 F Power of Fibonacci --数学+逆元+快速幂
原题:ZOJ 3774 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3774 --------------------- ...
- 2014 Super Training #6 F Search in the Wiki --集合取交+暴力
原题: ZOJ 3674 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3674 题意不难理解,很容易想到用暴力,但是无从下 ...
- 2014 Super Training #2 F The Bridges of Kolsberg --DP
原题:UVA 1172 http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_ ...
- 2014 Super Training #1 F Passage 概率DP
原题: HDU 3366 http://acm.hdu.edu.cn/showproblem.php?pid=3366 本来用贪心去做,怎么都WA,后来看网上原来是一个DP题. 首先按P/Q来做排 ...
- 2014 Super Training #10 C Shadow --SPFA/随便搞/DFS
原题: FZU 2169 http://acm.fzu.edu.cn/problem.php?pid=2169 这题貌似有两种解法,DFS和SPFA,但是DFS怎么都RE,SPFA也要用邻接表表示边, ...
- ZOJ 3686 A Simple Tree Problem
A Simple Tree Problem Time Limit: 3 Seconds Memory Limit: 65536 KB Given a rooted tree, each no ...
- BNU 28887——A Simple Tree Problem——————【将多子树转化成线段树+区间更新】
A Simple Tree Problem Time Limit: 3000ms Memory Limit: 65536KB This problem will be judged on ZJU. O ...
随机推荐
- android.widget.RadioButton 单选按钮(转)
大家好,我们今天这一节要介绍的是RadioGroup 的组事件.RadioGroup 可将各自不同的RadioButton ,设限于同一个Radio 按钮组,同一个RadioGroup 组里的按钮,只 ...
- .NET WinForm画树叶小程序
看了一片文章(http://keleyi.com/a/bjac/nurox416.htm),是使用分型画树叶,代码是Java的,因为Java很久没弄了,改用C#实现,下载地址: 画树叶小程序下载 核心 ...
- XStream的例子
写个小例子,方便以后复习: 1 package com.demo; 2 3 import java.util.ArrayList; 4 import java.util.List; 5 6 impor ...
- CentOS6.5上编译OpenJDK7源码
本文地址:http://www.cnblogs.com/wuyudong/p/build-openjdk7.html,转载请注明源地址. 采用开源的OpenJDK版本,获取其源码的方式有两种: 通Me ...
- Android ListView用EditText实现搜索功能
前言 最近在开发一个IM项目的时候有一个需求就是,好友搜索功能.即在EditText中输入好友名字,ListView列表中动态展示刷选的好友列表.我把这个功能抽取出来了,先贴一下效果图: 分析 在查阅 ...
- 自定义控件--CircleImageView(类似于QQ、微信圆形头像自定义控件)
现在基本上所有的需要用户注册的APP都有一个需要用户上传头像的需求,上传的头像基本都是类似于QQ.微信等社交应用圆形头像.最近,正在做的一个社交应用多处需要用到这种圆形头像的处理,总不能每次都对图片做 ...
- 可折叠的ToolBar+抽屉菜单NavigationView+浮动按钮FloatButton
使用Material Design风格的ToolBar和抽屉导航 先看个简单的运行效果 主要记录下布局的写法 1 用到的Google Design依赖和V7包依赖 compile 'com.andro ...
- 如何创建可扩展表视图中的iOS 学习和拓展优化(有待更新)
首先介绍老外的文章:<How To Create an Expandable Table View in iOS>这是老外用Swift实现 的,对应的老外github项目源码:https: ...
- 导出excel乱码问题
今天遇到一个问题,在用C#做导出excel的时候,出现excel乱码问题.百度了下. 发现问题如下: 非中文字符编码问题. 解决方法: 把输出的excel格式设置成UTF-8即可. 更改代码: Res ...
- 敏捷开发中高质量 Java 代码开发实践
Java 项目开发过程中,由于开发人员的经验.代码风格各不相同,以及缺乏统一的标准和管理流程,往往导致整个项目的代码质量较差,难于维护,需要较大的测试投入 和周期等问题. 这些问题在一个项目组初建.需 ...