【BZOJ3083】遥远的国度

Description

描述
zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度。当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn完成任务后才能进入遥远的国度继续追杀。

问题是这样的:遥远的国度有n个城市,这些城市之间由一些路连接且这些城市构成了一颗树。这个国度有一个首都,我们可以把这个首都看做整棵树的根,但遥远的国度比较奇怪,首都是随时有可能变为另外一个城市的。遥远的国度的每个城市有一个防御值,有些时候RapiD会使得某两个城市之间的路径上的所有城市的防御值都变为某个值。RapiD想知道在某个时候,如果把首都看做整棵树的根的话,那么以某个城市为根的子树的所有城市的防御值最小是多少。由于RapiD无法解决这个问题,所以他拦住了zcwwzdjn希望他能帮忙。但zcwwzdjn还要追杀sb的zhx,所以这个重大的问题就被转交到了你的手上。

Input

第1行两个整数n m,代表城市个数和操作数。
第2行至第n行,每行两个整数 u v,代表城市u和城市v之间有一条路。
第n+1行,有n个整数,代表所有点的初始防御值。
第n+2行一个整数 id,代表初始的首都为id。
第n+3行至第n+m+2行,首先有一个整数opt,如果opt=1,接下来有一个整数id,代表把首都修改为id;如果opt=2,接下来有三个整数p1 p2 v,代表将p1 p2路径上的所有城市的防御值修改为v;如果opt=3,接下来有一个整数 id,代表询问以城市id为根的子树中的最小防御值。

Output

对于每个opt=3的操作,输出一行代表对应子树的最小点权值。

Sample Input

3 7
1 2
1 3
1 2 3
1
3 1
2 1 1 6
3 1
2 2 2 5
3 1
2 3 3 4
3 1

Sample Output

1
2
3
4
提示
对于20%的数据,n<=1000 m<=1000。
对于另外10%的数据,n<=100000,m<=100000,保证修改为单点修改。
对于另外10%的数据,n<=100000,m<=100000,保证树为一条链。
对于另外10%的数据,n<=100000,m<=100000,没有修改首都的操作。
对于100%的数据,n<=100000,m<=100000,0<所有权值<=2^31。

题解:权值<=(1<<31)吓人~

一看到本题空间1280MB,第一反应就是先开它个2亿的数组再说~

如果没有换根操作,直接树剖线段树水过,但是就算有换根操作用树剖也是能搞的。

设当前根为root,想要查询x的子树中的最小值。考虑在原树中,如果x与root重合,那么答案就是整棵树的最小值;如果当前的root不在x原来的子树中,那么x现在的子树就是原来的子树,直接正常搞就行了。如果root在x原来的子树中,我们设root在x的ch儿子的子树中,发现x现在的子树就是(整棵树-ch原来的子树),那么我们需要知道ch的位置,此时又需要分类讨论了。

如果你没太看懂上面的叙述,可以通过看下面的图来理解一下。

这种情况显然直接搞

这种情况我们要讨论

我们看一下从root往上跳,在跳到x之前的最后一步,如果最后一步在重链上,那么ch的DFS序就是x的DFS序+1

如果最后一步在轻链上,那么跳最后一步之前的那个点就是ch

BZOJ3083

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <iostream>
  4. #define lson x<<1
  5. #define rson x<<1|1
  6. using namespace std;
  7. const int maxn=100010;
  8. int n,m,cnt,root;
  9. typedef unsigned int ll;
  10. int to[maxn<<1],next[maxn<<1],head[maxn],dep[maxn],fa[maxn],son[maxn],top[maxn],p[maxn],q[maxn],v[maxn];
  11. int siz[maxn],Q[maxn];
  12. ll s[maxn<<2],tag[maxn<<2];
  13. void pushdown(int x)
  14. {
  15. if(tag[x]) s[lson]=tag[lson]=s[rson]=tag[rson]=tag[x],tag[x]=0;
  16. }
  17. void updata(int l,int r,int x,int a,int b,ll c)
  18. {
  19. if(a<=l&&r<=b)
  20. {
  21. s[x]=tag[x]=c;
  22. return ;
  23. }
  24. pushdown(x);
  25. int mid=l+r>>1;
  26. if(a<=mid) updata(l,mid,lson,a,b,c);
  27. if(b>mid) updata(mid+1,r,rson,a,b,c);
  28. s[x]=min(s[lson],s[rson]);
  29. }
  30. ll query(int l,int r,int x,int a,int b)
  31. {
  32. if(a>b) return ((ll)1)<<31;
  33. if(a<=l&&r<=b) return s[x];
  34. pushdown(x);
  35. int mid=l+r>>1;
  36. if(b<=mid) return query(l,mid,lson,a,b);
  37. if(a>mid) return query(mid+1,r,rson,a,b);
  38. return min(query(l,mid,lson,a,b),query(mid+1,r,rson,a,b));
  39. }
  40. ll rd()
  41. {
  42. ll ret=0; char gc=getchar();
  43. while(gc<'0'||gc>'9') gc=getchar();
  44. while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
  45. return ret;
  46. }
  47. void dfs1(int x)
  48. {
  49. siz[x]=1;
  50. for(int i=head[x];i!=-1;i=next[i])
  51. {
  52. if(to[i]!=fa[x])
  53. {
  54. fa[to[i]]=x,dep[to[i]]=dep[x]+1,dfs1(to[i]),siz[x]+=siz[to[i]];
  55. if(siz[to[i]]>siz[son[x]]) son[x]=to[i];
  56. }
  57. }
  58. }
  59. void dfs2(int x,int tp)
  60. {
  61. top[x]=tp,p[x]=++p[0],Q[p[0]]=x;
  62. updata(1,n,1,p[0],p[0],v[x]);
  63. if(son[x]) dfs2(son[x],tp);
  64. for(int i=head[x];i!=-1;i=next[i])
  65. if(to[i]!=fa[x]&&to[i]!=son[x])
  66. dfs2(to[i],to[i]);
  67. q[x]=p[0];
  68. }
  69. void modify()
  70. {
  71. int x=rd(),y=rd();
  72. ll val=rd();
  73. while(top[x]!=top[y])
  74. {
  75. if(dep[top[x]]<dep[top[y]]) swap(x,y);
  76. updata(1,n,1,p[top[x]],p[x],val),x=fa[top[x]];
  77. }
  78. if(dep[x]>dep[y]) swap(x,y);
  79. updata(1,n,1,p[x],p[y],val);
  80. }
  81. void getans()
  82. {
  83. int x=rd(),y=root,z=root;
  84. if(x==root)
  85. {
  86. printf("%u\n",s[1]);
  87. return ;
  88. }
  89. while(top[y]!=top[x]&&y) z=top[y],y=fa[top[y]];
  90. if(dep[x]>dep[y])
  91. {
  92. printf("%u\n",query(1,n,1,p[x],q[x]));
  93. return ;
  94. }
  95. if(x!=y) z=Q[p[x]+1];
  96. printf("%u\n",min(query(1,n,1,1,p[z]-1),query(1,n,1,q[z]+1,n)));
  97. }
  98. void add(int a,int b)
  99. {
  100. to[cnt]=b,next[cnt]=head[a],head[a]=cnt++;
  101. }
  102. int main()
  103. {
  104. n=rd(),m=rd();
  105. int i,a,b;
  106. memset(head,-1,sizeof(head));
  107. for(i=1;i<n;i++) a=rd(),b=rd(),add(a,b),add(b,a);
  108. for(i=1;i<=n;i++) v[i]=rd();
  109. dep[1]=1,dfs1(1),dfs2(1,1);
  110. root=rd();
  111. for(i=1;i<=m;i++)
  112. {
  113. a=rd();
  114. if(a==1) root=rd();
  115. if(a==2) modify();
  116. if(a==3) getans();
  117. }
  118. return 0;
  119. }

BZOJ3306

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <iostream>
  4. #define lson x<<1
  5. #define rson x<<1|1
  6. using namespace std;
  7. const int maxn=100010;
  8. int n,m,cnt,root;
  9. int to[maxn<<1],next[maxn<<1],head[maxn],dep[maxn],fa[maxn],son[maxn],top[maxn],p[maxn],q[maxn],v[maxn];
  10. int siz[maxn],Q[maxn];
  11. int s[maxn<<2];
  12. char str[maxn];
  13. void updata(int l,int r,int x,int a,int b)
  14. {
  15. if(l==r)
  16. {
  17. s[x]=b;
  18. return ;
  19. }
  20. int mid=l+r>>1;
  21. if(a<=mid) updata(l,mid,lson,a,b);
  22. else updata(mid+1,r,rson,a,b);
  23. s[x]=min(s[lson],s[rson]);
  24. }
  25. int query(int l,int r,int x,int a,int b)
  26. {
  27. if(a>b) return 0x7fffffff;
  28. if(a<=l&&r<=b) return s[x];
  29. int mid=l+r>>1;
  30. if(b<=mid) return query(l,mid,lson,a,b);
  31. if(a>mid) return query(mid+1,r,rson,a,b);
  32. return min(query(l,mid,lson,a,b),query(mid+1,r,rson,a,b));
  33. }
  34. int rd()
  35. {
  36. int ret=0,f=1; char gc=getchar();
  37. while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();}
  38. while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();
  39. return ret*f;
  40. }
  41. void dfs1(int x)
  42. {
  43. siz[x]=1;
  44. for(int i=head[x];i!=-1;i=next[i])
  45. {
  46. if(to[i]!=fa[x])
  47. {
  48. fa[to[i]]=x,dep[to[i]]=dep[x]+1,dfs1(to[i]),siz[x]+=siz[to[i]];
  49. if(siz[to[i]]>siz[son[x]]) son[x]=to[i];
  50. }
  51. }
  52. }
  53. void dfs2(int x,int tp)
  54. {
  55. top[x]=tp,p[x]=++p[0],Q[p[0]]=x;
  56. updata(1,n,1,p[0],v[x]);
  57. if(son[x]) dfs2(son[x],tp);
  58. for(int i=head[x];i!=-1;i=next[i])
  59. if(to[i]!=fa[x]&&to[i]!=son[x])
  60. dfs2(to[i],to[i]);
  61. q[x]=p[0];
  62. }
  63. void getans()
  64. {
  65. int x=rd(),y=root,z=root;
  66. if(x==root)
  67. {
  68. printf("%d\n",s[1]);
  69. return ;
  70. }
  71. while(top[y]!=top[x]&&y) z=top[y],y=fa[top[y]];
  72. if(dep[x]>dep[y])
  73. {
  74. printf("%d\n",query(1,n,1,p[x],q[x]));
  75. return ;
  76. }
  77. if(x!=y) z=Q[p[x]+1];
  78. printf("%d\n",min(query(1,n,1,1,p[z]-1),query(1,n,1,q[z]+1,n)));
  79. }
  80. void add(int a,int b)
  81. {
  82. to[cnt]=b,next[cnt]=head[a],head[a]=cnt++;
  83. }
  84. int main()
  85. {
  86. n=rd(),m=rd();
  87. int i,a,b;
  88. memset(head,-1,sizeof(head));
  89. for(i=1;i<=n;i++)
  90. {
  91. fa[i]=rd(),v[i]=rd();
  92. if(!fa[i]) root=i;
  93. else add(fa[i],i);
  94. }
  95. dep[root]=1,dfs1(root),dfs2(root,root);
  96. for(i=1;i<=m;i++)
  97. {
  98. scanf("%s",str);
  99. if(str[0]=='V') a=rd(),b=rd(),updata(1,n,1,p[a],b);
  100. if(str[0]=='E') root=rd();
  101. if(str[0]=='Q') getans();
  102. }
  103. return 0;
  104. }

【BZOJ3083/3306】遥远的国度/树 树链剖分+线段树的更多相关文章

  1. 【bzoj3083】遥远的国度 树链剖分+线段树

    题目描述 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcwwzdjn ...

  2. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

  3. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  4. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  5. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  6. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

  7. HDU4897 (树链剖分+线段树)

    Problem Little Devil I (HDU4897) 题目大意 给定一棵树,每条边的颜色为黑或白,起始时均为白. 支持3种操作: 操作1:将a->b的路径中的所有边的颜色翻转. 操作 ...

  8. Aizu 2450 Do use segment tree 树链剖分+线段树

    Do use segment tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.bnuoj.com/v3/problem_show ...

  9. 【POJ3237】Tree(树链剖分+线段树)

    Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edg ...

  10. HDU 2460 Network(双连通+树链剖分+线段树)

    HDU 2460 Network 题目链接 题意:给定一个无向图,问每次增加一条边,问个图中还剩多少桥 思路:先双连通缩点,然后形成一棵树,每次增加一条边,相当于询问这两点路径上有多少条边,这个用树链 ...

随机推荐

  1. MySQL Win7 64位 下载、安装与配置图文教程

    MySQL是目前十分流行的一种关系型数据库管理系统.官网推出的安装包有两种格式,分别是:ZIP格式和MSI格式.其中MSI格式的可以直接点击安装,按照它给出的安装提示进行安装,一般MySQL将会安装在 ...

  2. Jacoco覆盖率工具使用之maven篇

    说明 之前的文章已经介绍过如何使用apacheant 执行jacoco工具,下面开始介绍如何使用maven使用jacoco工具. 1.首先新建一个maven项目       如图所示:        ...

  3. git学习——分支

    分支 创建分支:git branch 如:git branch testing Git通过HEAD指针知道用户是在哪一个分支上工作. 切换分支用git checkout命令,注意:可以用git sta ...

  4. 移动端H5页面 input 获取焦点时,虚拟键盘挡住input输入框解决方法

    在移动端h5开发的时候,发现如果input在页面底部,当触发input焦点的时候会弹出系统虚拟键盘,虚拟键盘会遮挡input输入框.这会很影响用户体验,于是在网上找到了如下的解决办法: 方法一:使用w ...

  5. Java 连接使用 Redis

    1. 开始在 Java 中使用 Redis 前, 我们需要确保已经安装了 redis 服务及 Java redis 驱动,且你的机器上能正常使用 Java. 首先你需要下载驱动包,下载 jedis.j ...

  6. C语言-十进制转换为二进制函数

    char * itobs(int num, char * str) { int i; * sizeof(int); ; i >= ; i--, num >>= ) { str[i] ...

  7. Bootstrap-初步学习

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="utf-8&quo ...

  8. 由"永恒之蓝"病毒而来的电脑知识科普

    永恒之蓝病毒事件: 继英国医院被攻击,随后在刚刚过去的5月12日晚上20点左右肆虐中国高校的WannaCry勒索事件,全国各地的高校学生纷纷反映,自己的电脑遭到病毒的攻击,文档被加密,壁纸遭到篡改,并 ...

  9. JS 自动计算HTML的font-size

    Rem尺寸解决方案,需要配合一些js动态设置<html>标签的font-size 和 viewport来配合 <script> (function(doc, win) { va ...

  10. silverlight RadGridView总结三(转载)

    在RadGridView中进行分组以及导出 分组 主要是在前台进行分组的定义: 前台代码: <telerik:RadGridView x:Name="RadGridView1" ...