Description

在2016年,佳媛姐姐刚刚学习了树,非常开心。

现在她想解决这样一个问题:给定一颗有根树(根为1),有以下两种操作:

1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标记,其他结点均无标记,而且对于某个结点,可以打多次标记);

2. 询问操作:询问某个结点最近的一个打了标记的祖先(这个结点本身也算自己的祖

先).

你能帮帮她吗?

Input

输入第一行两个正整数,分别表示节点个数和操作次数.

接下来行,每行两个正整数,表示有一条有向边.

接下来行,形如“”.为“”时,表示这是一个标记操作;

为“”时,表示这是一个询问操作.

Output

输出一个正整数,表示结果.

Sample Input

5 5
1 2
1 3
2 4
2 5
Q 2
C 2
Q 2
Q 5
Q 3

Sample Output

1
2
2
1

HINT

Solution

正难则反,对于所有操作逆序离线处理.

先记录每个节点被打过几次标记,预处理出每个节点最后时刻最近的打了标记的祖先.

从后往前消除标记,如果当前点的标记被消除完,则这个点最近的打了标记的祖先为它父亲最近的打了标记的祖先,并查集可实现.

  1. #include<cmath>
  2. #include<ctime>
  3. #include<stack>
  4. #include<queue>
  5. #include<cstdio>
  6. #include<vector>
  7. #include<cstring>
  8. #include<cstdlib>
  9. #include<iostream>
  10. #include<algorithm>
  11. #define N 100005
  12. using namespace std;
  13. struct graph{
  14. int nxt,to;
  15. }e[N<<1];
  16. struct quest{
  17. int t,x;
  18. }b[N];
  19. int a[N],f[N],g[N],fa[N],ans[N],n,m,q,t,x,cnt;
  20. stack<int> s;
  21. inline int read(){
  22. int ret=0;char c=getchar();
  23. while(!isdigit(c)) c=getchar();
  24. while(isdigit(c)){
  25. ret=(ret<<1)+(ret<<3)+c-'0';
  26. c=getchar();
  27. }
  28. return ret;
  29. }
  30. inline int rc(){
  31. char c=getchar();
  32. while(c!='Q'&&c!='C') c=getchar();
  33. if(c=='C') return 1;
  34. return 2;
  35. }
  36. inline void addedge(int x,int y){
  37. e[++cnt].nxt=g[x];g[x]=cnt;e[cnt].to=y;
  38. }
  39. inline void dfs(int u){
  40. fa[u]=f[u]=1;s.push(u);
  41. while(!s.empty()){
  42. u=s.top();s.pop();
  43. for(int i=g[u];i;i=e[i].nxt)
  44. if(fa[u]!=e[i].to){
  45. fa[e[i].to]=u;s.push(e[i].to);
  46. if(!a[e[i].to]) f[e[i].to]=f[u];
  47. else f[e[i].to]=e[i].to;
  48. }
  49. }
  50. }
  51. inline int gf(int k){
  52. if(f[k]==k) return k;
  53. return f[k]=gf(f[k]);
  54. }
  55. inline void init(){
  56. n=read();q=read();
  57. for(int i=1,j,k;i<n;++i){
  58. j=read();k=read();
  59. addedge(j,k);addedge(k,j);
  60. }
  61. for(int i=1;i<=q;++i){
  62. b[i].t=rc();b[i].x=read();
  63. if(b[i].t&1) ++a[b[i].x];
  64. }
  65. ++a[1];dfs(1);cnt=0;
  66. for(int i=q,j,k;i;--i){
  67. if(b[i].t&1){
  68. if(!(--a[b[i].x])){
  69. f[b[i].x]=gf(f[fa[b[i].x]]);
  70. }
  71. }
  72. else ans[++cnt]=gf(f[b[i].x]);
  73. }
  74. for(int i=cnt;i;--i)
  75. printf("%d\n",ans[i]);
  76. }
  77. int main(){
  78. freopen("tree.in","r",stdin);
  79. freopen("tree.out","w",stdout);
  80. init();
  81. fclose(stdin);
  82. fclose(stdout);
  83. return 0;
  84. }

[bzoj4551][Tjoi2016][Heoi2016]树的更多相关文章

  1. [BZOJ4551][TJOI2016&&HEOI2016]树(并查集)

    4551: [Tjoi2016&Heoi2016]树 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1746  Solved: 800[Sub ...

  2. BZOJ4551 Tjoi2016&Heoi2016树(离线+并查集)

    似乎是弱化的qtree3.树剖什么的非常无脑.考虑离线.并查集维护每个点的最近打标记祖先,倒序处理,删除标记时将其与父亲合并即可. #include<iostream> #include& ...

  3. BZOJ4551——[Tjoi2016&Heoi2016]树

    1.题意: 给定一颗有根树(根为1),有以下 两种操作:1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标记,其他结点均无标记,而且对于某个 结点,可以打多次标记.)2. 询问操作:询问某个 ...

  4. BZOJ4551: [Tjoi2016&Heoi2016]树

    Description 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为1),有以下 两种操作:1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标 ...

  5. [bzoj4551][Tjoi2016&Heoi2016]树-树链剖分

    Brief Description 给定一颗有根树(根为1),有以下 两种操作:1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标记,其他结点均无标记,而且对于某个 结点,可以打多次标记.) ...

  6. BZOJ4551[Tjoi2016&Heoi2016]树——dfs序+线段树/树链剖分+线段树

    题目描述 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为1),有以下 两种操作:1. 标记操作:对某个结点打上标记(在最开始,只有结点1有标记,其他结点均 ...

  7. 【BZOJ4551】[Tjoi2016&Heoi2016]树 并查集

    [BZOJ4551][Tjoi2016&Heoi2016]树 Description 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为1),有以下两 ...

  8. BZOJ 4551: [Tjoi2016&Heoi2016]树

    4551: [Tjoi2016&Heoi2016]树 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 748  Solved: 394[Subm ...

  9. BZOJ_4551_[Tjoi2016&Heoi2016]树_树剖+线段树

    BZOJ_4551_[Tjoi2016&Heoi2016]树_树剖+线段树 Description 在2016年,佳媛姐姐刚刚学习了树,非常开心.现在他想解决这样一个问题:给定一颗有根树(根为 ...

随机推荐

  1. 安装ESXi5.5遇到Relocating modules and starting up the kernel的处理

    在一些Dell较旧的服务器上安装ESXi 5.x时, 会遇到卡在Relocating modules and starting up the kernel过不去的问题. 比如我装的这台CS24VSS. ...

  2. 正在编译转换: 未能找到元数据文件 EntityFramework.dll

    错误 1 正在编译转换: 未能找到元数据文件“C:\Program Files (x86)\Microsoft Visual Studio 11.0\Common7\Tools\..\IDE\Enti ...

  3. Splay整理

    伸展树(Splay Tree),也叫分裂树,是一种二叉排序树,它能在O(log n)内完成插入.查找和删除操作.(来自百科) 伸展树的操作主要是

  4. COGS 2. 旅行计划

    2. 旅行计划 ★☆   输入文件:djs.in   输出文件:djs.out   简单对比时间限制:3 s   内存限制:128 MB 过暑假了,阿杜准备出行旅游,他已经查到了某些城市的两两之间的距 ...

  5. MVC控制器总结

    1.特性 [AuthorizeFilter]  用于权限过滤 [HttpGet] [HttpPost] 2.参数 获取方法 public ActionResult void Get(int id){} ...

  6. 表单验证——JavaScript和Jquery版

    1.轻量级的JavaScript表单验证 在应用中引用 validator.min.js 文件 <script type="text/javascript" src=&quo ...

  7. Vue系列: 如何通过组件的属性props设置样式

    比如我们要在vue中显示百度地图,然后将相关的代码包装成组件,然后需要由外部来设置组件的高度,关于props的介绍,可以参考: http://cn.vuejs.org/guide/components ...

  8. Bootstrap系列 -- 12. 水平表单

    Bootstrap框架默认的表单是垂直显示风格,但很多时候我们需要的水平表单风格(标签居左,表单控件居右) 在Bootstrap框架中要实现水平表单效果,必须满足以下两个条件: 1.在<form ...

  9. Windows Azure 云服务角色架构

    当我们使用VS发布一个Cloud Service或者在Portal上上传发布包后,就能启动和运行一个云服务,可以保护WebRole,WorkerRole的一个或者多个实例. Windows Azure ...

  10. centos7.2安装phpmyadmin

    首先服务器要有web 环境 yum install phpmyadmin 修改/etc/http.conf/conf.d/phpMyadmin.conf 将  #Require ip 127.0.0. ...