洛谷

Codeforces


思路

首先要莫名其妙地想到圆方树。

建起圆方树后,令方点的权值是双联通分量中的最小值,那么\((u,v)\)的答案就是路径\((u,v)\)上的最小值。

然而这题还有修改,可以在每个方点维护一个\(multiset\)以支持。

但如果每次修改都暴力修改相邻的方点权值显然要挂,如何优化?

可以令方点的权值不包括自己的父亲,那么修改圆点时只需要修改自己的父亲即可。

查询时如果\(lca\)是方点那么还要与方点的父亲取\(\min\)。

比较码农。


代码

  1. #include<bits/stdc++.h>
  2. clock_t t=clock();
  3. namespace my_std{
  4. using namespace std;
  5. #define pii pair<int,int>
  6. #define fir first
  7. #define sec second
  8. #define MP make_pair
  9. #define rep(i,x,y) for (int i=(x);i<=(y);i++)
  10. #define drep(i,x,y) for (int i=(x);i>=(y);i--)
  11. #define go(x) for (int i=head[x];i;i=edge[i].nxt)
  12. #define templ template<typename T>
  13. #define sz 201001
  14. typedef long long ll;
  15. typedef double db;
  16. mt19937 rng(chrono::steady_clock::now().time_since_epoch().count());
  17. templ inline T rnd(T l,T r) {return uniform_int_distribution<T>(l,r)(rng);}
  18. templ inline bool chkmax(T &x,T y){return x<y?x=y,1:0;}
  19. templ inline bool chkmin(T &x,T y){return x>y?x=y,1:0;}
  20. templ inline void read(T& t)
  21. {
  22. t=0;char f=0,ch=getchar();double d=0.1;
  23. while(ch>'9'||ch<'0') f|=(ch=='-'),ch=getchar();
  24. while(ch<='9'&&ch>='0') t=t*10+ch-48,ch=getchar();
  25. if(ch=='.'){ch=getchar();while(ch<='9'&&ch>='0') t+=d*(ch^48),d*=0.1,ch=getchar();}
  26. t=(f?-t:t);
  27. }
  28. template<typename T,typename... Args>inline void read(T& t,Args&... args){read(t); read(args...);}
  29. char __sr[1<<21],__z[20];int __C=-1,__Z=0;
  30. inline void __Ot(){fwrite(__sr,1,__C+1,stdout),__C=-1;}
  31. inline void print(register int x)
  32. {
  33. if (__C>1<<20) __Ot(); if (x<0) __sr[++__C]='-',x=-x;
  34. while (__z[++__Z]=x%10+48,x/=10);
  35. while (__sr[++__C]=__z[__Z],--__Z);__sr[++__C]='\n';
  36. }
  37. void file()
  38. {
  39. #ifndef ONLINE_JUDGE
  40. freopen("a.in","r",stdin);
  41. #endif
  42. }
  43. inline void chktime()
  44. {
  45. #ifndef ONLINE_JUDGE
  46. cout<<(clock()-t)/1000.0<<'\n';
  47. #endif
  48. }
  49. #ifdef mod
  50. ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x%mod) if (y&1) ret=ret*x%mod;return ret;}
  51. ll inv(ll x){return ksm(x,mod-2);}
  52. #else
  53. ll ksm(ll x,int y){ll ret=1;for (;y;y>>=1,x=x*x) if (y&1) ret=ret*x;return ret;}
  54. #endif
  55. // inline ll mul(ll a,ll b){ll d=(ll)(a*(double)b/mod+0.5);ll ret=a*b-d*mod;if (ret<0) ret+=mod;return ret;}
  56. }
  57. using namespace my_std;
  58. int n,m,Q,N;
  59. int val[sz];
  60. multiset<int>s[sz];
  61. struct hh{int t,nxt;}edge[sz<<1];
  62. int head[sz],ecnt;
  63. void make_edge(int f,int t)
  64. {
  65. edge[++ecnt]=(hh){t,head[f]};
  66. head[f]=ecnt;
  67. edge[++ecnt]=(hh){f,head[t]};
  68. head[t]=ecnt;
  69. }
  70. namespace BuildTree
  71. {
  72. struct hh{int t,nxt;}edge[sz<<1];
  73. int head[sz],ecnt;
  74. void make_edge(int f,int t)
  75. {
  76. edge[++ecnt]=(hh){t,head[f]};
  77. head[f]=ecnt;
  78. edge[++ecnt]=(hh){f,head[t]};
  79. head[t]=ecnt;
  80. }
  81. int dfn[sz],low[sz],cnt;
  82. stack<int>s;
  83. bool in[sz];
  84. void tarjan(int x,int fa)
  85. {
  86. dfn[x]=low[x]=++cnt;in[x]=1;s.push(x);
  87. #define v edge[i].t
  88. go(x) if (v!=fa)
  89. {
  90. if (!dfn[v])
  91. {
  92. tarjan(v,x),chkmin(low[x],low[v]);
  93. if (low[v]>=dfn[x])
  94. {
  95. ++N;::make_edge(x,N);
  96. int y;
  97. do{y=s.top();s.pop();in[y]=0;::make_edge(y,N);}while (y!=v);
  98. }
  99. }
  100. else chkmin(low[x],dfn[v]);
  101. }
  102. #undef v
  103. }
  104. void init()
  105. {
  106. read(n,m,Q);N=n;
  107. rep(i,1,n) read(val[i]);
  108. int x,y;
  109. rep(i,1,m) read(x,y),make_edge(x,y);
  110. tarjan(1,0);
  111. }
  112. }
  113. int dfn[sz],pre[sz],top[sz],fa[sz],dep[sz],size[sz],son[sz],cnt;
  114. #define v edge[i].t
  115. void dfs1(int x,int fa)
  116. {
  117. dep[x]=dep[::fa[x]=fa]+1;size[x]=1;
  118. if (x<=n&&x!=1) s[fa].insert(val[x]);
  119. go(x) if (v!=fa)
  120. {
  121. dfs1(v,x);
  122. size[x]+=size[v];
  123. if (size[v]>size[son[x]]) son[x]=v;
  124. }
  125. if (x>n) val[x]=*s[x].begin();
  126. }
  127. void dfs2(int x,int fa,int tp)
  128. {
  129. top[pre[dfn[x]=++cnt]=x]=tp;
  130. if (son[x]) dfs2(son[x],x,tp);
  131. go(x) if (v!=fa&&v!=son[x]) dfs2(v,x,v);
  132. }
  133. #undef v
  134. int mn[sz<<2];
  135. #define ls k<<1
  136. #define rs k<<1|1
  137. #define lson ls,l,mid
  138. #define rson rs,mid+1,r
  139. void pushup(int k){mn[k]=min(mn[ls],mn[rs]);}
  140. void modify(int k,int l,int r,int x,int y)
  141. {
  142. if (l==r) return void(mn[k]=y);
  143. int mid=(l+r)>>1;
  144. if (x<=mid) modify(lson,x,y);
  145. else modify(rson,x,y);
  146. pushup(k);
  147. }
  148. int query(int k,int l,int r,int x,int y)
  149. {
  150. if (x<=l&&r<=y) return mn[k];
  151. int mid=(l+r)>>1,ret=1e9;
  152. if (x<=mid) chkmin(ret,query(lson,x,y));
  153. if (y>mid) chkmin(ret,query(rson,x,y));
  154. return ret;
  155. }
  156. #undef ls
  157. #undef rs
  158. #undef lson
  159. #undef rson
  160. void modify(int x,int w)
  161. {
  162. if (x==1) return void(modify(1,1,N,dfn[x],val[x]=w));
  163. s[fa[x]].erase(s[fa[x]].find(val[x]));
  164. s[fa[x]].insert(w);
  165. val[x]=w;
  166. modify(1,1,N,dfn[x],w);
  167. modify(1,1,N,dfn[fa[x]],*s[fa[x]].begin());
  168. }
  169. int query(int x,int y)
  170. {
  171. int ret=1e9;
  172. while (top[x]!=top[y])
  173. {
  174. if (dep[top[x]]<dep[top[y]]) swap(x,y);
  175. chkmin(ret,query(1,1,N,dfn[top[x]],dfn[x]));
  176. x=fa[top[x]];
  177. }
  178. if (dep[x]>dep[y]) swap(x,y);
  179. chkmin(ret,query(1,1,N,dfn[x],dfn[y]));
  180. if (x!=1&&x>n) chkmin(ret,val[fa[x]]);
  181. return ret;
  182. }
  183. int main()
  184. {
  185. file();
  186. BuildTree::init();
  187. dfs1(1,0);dfs2(1,0,1);
  188. rep(i,1,N) modify(1,1,N,dfn[i],val[i]);
  189. char c;int x,y;
  190. while (Q--)
  191. {
  192. c=getchar();while (c!='C'&&c!='A') c=getchar();
  193. read(x,y);
  194. if (c=='C') modify(x,y);
  195. else printf("%d\n",query(x,y));
  196. }
  197. return 0;
  198. }

Codeforces 487E Tourists [广义圆方树,树链剖分,线段树]的更多相关文章

  1. CodeForces 487E Tourists(圆方树+线段树+树链剖分)

    题意 ​ \(n\) 个点 \(m\) 条边的无向连通图,每个点有点权,\(q\) 个要求,每次更新一个点的点权或查询两点间路径权值最小的点最小的路径. 思路 ​ 算是圆方树的板子吧?圆方树处理的主要 ...

  2. Water Tree CodeForces 343D 树链剖分+线段树

    Water Tree CodeForces 343D 树链剖分+线段树 题意 给定一棵n个n-1条边的树,起初所有节点权值为0. 然后m个操作, 1 x:把x为根的子树的点的权值修改为1: 2 x:把 ...

  3. BZOJ2325[ZJOI2011]道馆之战——树链剖分+线段树

    题目描述 口袋妖怪(又名神奇宝贝或宠物小精灵)红/蓝/绿宝石中的水系道馆需要经过三个冰地才能到达馆主的面前,冰地中 的每一个冰块都只能经过一次.当一个冰地上的所有冰块都被经过之后,到下一个冰地的楼梯才 ...

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

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

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

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

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

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

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

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

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

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

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

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

随机推荐

  1. 《Python数据可视化编程实战》

    第一章:准备工作环境 WinPython-32bit-3.5.2.2Qt5.exe 1.1 设置matplotlib参数 配置模板以方便各项目共享 D:\Bin\WinPython-32bit-3.5 ...

  2. JavaScript 小工具

    1. 字符串格式化输出 支持形如: Orders of {1} or more {0}' {0},{1}代表第几个参数,包含了完善的异常处理.当给定参数少于格式化串中占位符个数时,未找到的直接留白. ...

  3. aplication.properties配置

    1.设置使用的properties文件 spring.profiles.active=dev 设置激活使用哪个properties一般设置两个,一个是开发环境的,一个是本地的测试环境 可设置默认使用开 ...

  4. 【tmos】mvn package相关知识点(待补充...)

    SpringBoot项目打包跳过测试 <build> <plugins> <plugin> <groupId>org.springframework.b ...

  5. Tomcat/7.0.81 远程代码执行漏洞复现

    Tomcat/7.0.81 远程代码执行漏洞复现 参考链接: http://www.freebuf.com/vuls/150203.html 漏洞描述: CVE-2017-12617 Apache T ...

  6. 关于Sublime Text 3的几个问题总结

    问题1:Sublime Text 3的注册码. 注册码网上搜有很多,所以可以去网上找.我现在给的可能以后就不能用了,而且我也是网上找的... 这个现在最新版本是可用的. —– BEGIN LICENS ...

  7. vue 学习笔记—Resource

    1.首先是引入  或者用npm来安装  cnpm i vue-resource --save(推荐) 3.提供的api 关于请求写法: get(){ // get请求 this.$http.get( ...

  8. vue 学习笔记—路由篇

    一.关于三种路由 动态路由 就是path:good/:ops    这种 用 $route.params接收 <router-link>是用来跳转  <router-view> ...

  9. 20165237 2017-2018-2 《Java程序设计》第十周考试补做及编程题

    20165237 2017-2018-2 <Java程序设计>第十周考试补做及编程题 知识点 1.链表是由若干个称作节点的对象组成的一种数据结构,每个节点含有一个数据和下一个节点的引用 . ...

  10. yum upgrade卡在 清理initial-setup-0.3.9.30-1.el7.centos.x86_64

    我安装CENTOS7.2,用yum -y update进行更新 卡在这里了 清理 : initial-setup-0.3.9.30-1.el7.cent 目测是一个系统bug,执行关闭命令解决: sy ...