圆方树简介(UOJ30:CF Round #278 Tourists)
我写这篇博客的原因
证明我也是学过圆方树的
顺便存存代码
前置技能
双联通分量:点双
然后就没辣
圆方树
建立
新建一个图
定义原图中的所有点为圆点
对于每个点双联通分量(只有两个点的也算)
建立一个方点,向所有的点双内的点连边
性质
- 一定是个森林
- 每个点双有唯一的方点
- 圆点方点相间分布,相同点不相邻
等等
例子 1
题面
求可以出现在两点之间的简单路路径上的点的最大权值,不带修改
分析
考虑用圆方树来解决
设圆点权值为本身,方点权值为点双中的最大权值
那么就是树上的路径最大权值
例子 2
还是上面的题,可以修改一个点的权值
类似题UOJ
用老方法
每次修改时更改圆点连接的所有方点
没了?
不存在的,这样每次修改是\(O(n)\)的,容易被卡
换一种定义:方点权值不包括它的父亲圆点
那么每次修改就只要修改圆点的父亲
注意如果\(lca\)是方点,还要算上它父亲方点的权值
堆+线段树(zkw辣)+树剖+圆方树+tarjan
UOJ代码
# include <bits/stdc++.h>
# define RG register
# define IL inline
# define Fill(a, b) memset(a, b, sizeof(a))
using namespace std;
typedef long long ll;
const int _(4e5 + 5);
IL int Input(){
RG int x = 0, z = 1; RG char c = getchar();
for(; c < '0' || c > '9'; c = getchar()) z = c == '-' ? -1 : 1;
for(; c >= '0' && c <= '9'; c = getchar()) x = (x << 1) + (x << 3) + (c ^ 48);
return x * z;
}
struct Edge{
int next[_], first[_], to[_], cnt;
IL void Init(){
Fill(first, -1);
}
IL void Add(RG int u, RG int v){
next[cnt] = first[u], to[cnt] = v, first[u] = cnt++;
}
} G1, G2;
struct Segment{
int mn[_ << 2], M;
IL void Init(RG int n){
Fill(mn, 127);
for(M = 1; M < n; M <<= 1);
}
IL void Update(RG int x, RG int y){
x += M - 1, mn[x] = y;
for(x >>= 1; x; x >>= 1) mn[x] = min(mn[x << 1], mn[x << 1 | 1]);
}
IL int Query(RG int l, RG int r){
RG int ret = 2e9;
for(l += M - 2, r += M; l ^ r ^ 1; l >>= 1, r >>= 1){
if(~l & 1) ret = min(ret, mn[l ^ 1]);
if(r & 1) ret = min(ret, mn[r ^ 1]);
}
return ret;
}
} T;
struct Heap{
priority_queue <int> Q1, Q2;
IL void Push(RG int x){
Q1.push(-x);
}
IL void Del(RG int x){
Q2.push(-x);
}
IL int Top(){
while(!Q2.empty() && Q1.top() == Q2.top()) Q1.pop(), Q2.pop();
return -Q1.top();
}
} Q[_];
int tmp, n, m, q, val[_], dfn[_], low[_], Index, S[_];
int size[_], top[_], fa[_], deep[_], son[_];
IL void Tarjan(RG int u){
dfn[u] = low[u] = ++Index, S[++S[0]] = u;
for(RG int e = G1.first[u]; e != -1; e = G1.next[e]){
RG int v = G1.to[e], x;
if(!dfn[v]){
Tarjan(v), low[u] = min(low[u], low[v]);
if(low[v] >= dfn[u]){
val[++n] = 2e9, x = 0;
do{
x = S[S[0]--];
G2.Add(n, x), G2.Add(x, n);
} while(x != v);
G2.Add(n, u), G2.Add(u, n);
}
}
else low[u] = min(low[u], dfn[v]);
}
}
IL void Dfs1(RG int u){
size[u] = 1;
if(u <= tmp && fa[u]) Q[fa[u]].Push(val[u]);
for(RG int e = G2.first[u]; e != -1; e = G2.next[e]){
RG int v = G2.to[e];
if(size[v]) continue;
fa[v] = u, deep[v] = deep[u] + 1;
Dfs1(v);
size[u] += size[v];
if(size[v] > size[son[u]]) son[u] = v;
}
}
IL void Dfs2(RG int u, RG int Top){
dfn[u] = ++Index, top[u] = Top;
if(son[u]) Dfs2(son[u], Top);
for(RG int e = G2.first[u]; e != -1; e = G2.next[e])
if(!dfn[G2.to[e]]) Dfs2(G2.to[e], G2.to[e]);
}
IL int Query(RG int u, RG int v){
RG int ret = 2e9;
while(top[u] ^ top[v]){
if(deep[top[u]] > deep[top[v]]) swap(u, v);
ret = min(ret, T.Query(dfn[top[v]], dfn[v]));
v = fa[top[v]];
}
if(dfn[u] > dfn[v]) swap(u, v);
ret = min(ret, T.Query(dfn[u], dfn[v]));
if(u > tmp) ret = min(ret, val[fa[u]]);
return ret;
}
int main(RG int argc, RG char* argv[]){
G1.Init(), G2.Init();
tmp = n = Input(), m = Input(), q = Input();
for(RG int i = 1; i <= n; ++i) val[i] = Input();
for(RG int i = 1; i <= m; ++i){
RG int u = Input(), v = Input();
G1.Add(u, v), G1.Add(v, u);
}
for(RG int i = 1; i <= tmp; ++i) if(!dfn[i]) Tarjan(i);
Fill(dfn, 0), Index = 0, T.Init(n);
Dfs1(1), Dfs2(1, 1);
for(RG int i = 1; i <= n; ++i) T.Update(dfn[i], val[i]);
for(RG int i = tmp + 1; i <= n; ++i) T.Update(dfn[i], Q[i].Top());
for(RG int i = 1, a, b; i <= q; ++i){
RG char op; scanf(" %c", &op);
a = Input(), b = Input();
if(op == 'C'){
if(fa[a]) Q[fa[a]].Del(val[a]);
val[a] = b, T.Update(dfn[a], val[a]);
if(fa[a]) Q[fa[a]].Push(val[a]);
if(fa[a]) T.Update(dfn[fa[a]], Q[fa[a]].Top());
}
else printf("%d\n", Query(a, b));
}
return 0;
}
圆方树简介(UOJ30:CF Round #278 Tourists)的更多相关文章
- UOJ #30. [CF Round #278] Tourists
UOJ #30. [CF Round #278] Tourists 题目大意 : 有一张 \(n\) 个点, \(m\) 条边的无向图,每一个点有一个点权 \(a_i\) ,你需要支持两种操作,第一种 ...
- 圆方树总结 [uoj30]Tourists
圆方树总结 所谓圆方树就是把一张图变成一棵树. 怎么变啊qaq 这里盗一张图 简单来说就是给每一个点双新建一个点,然后连向这个点双中的每一个点.特殊的,把两个点互相连通的也视作一个点双. 我们把原来就 ...
- [CF Round #278] Tourists
给定一个n个点m条边的无向图,求图上的两点的所有的简单路径之间的最小边. 蓝链 $ n,m,q \leq 100000, w_i \leq 10 ^7$ Solution 考虑建立用缩点双来建立广义圆 ...
- uoj30【CF Round #278】Tourists(圆方树+树链剖分+可删除堆)
- 学习了一波圆方树 学习了一波点分治 学习了一波可删除堆(巧用 ? STL) 传送门: Icefox_zhx 注意看代码看怎么构建圆方树的. tips:tips:tips:圆方树内存记得开两倍 CO ...
- UOJ#30/Codeforces 487E Tourists 点双连通分量,Tarjan,圆方树,树链剖分,线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ30.html 题目传送门 - UOJ#30 题意 uoj写的很简洁.清晰,这里就不抄一遍了. 题解 首先建 ...
- 仙人掌 && 圆方树 && 虚树 总结
仙人掌 && 圆方树 && 虚树 总结 Part1 仙人掌 定义 仙人掌是满足以下两个限制的图: 图完全联通. 不存在一条边处在两个环中. 其中第二个限制让仙人掌的题做 ...
- BZOJ5329:[SDOI2018]战略游戏(圆方树,虚树)
Description 省选临近,放飞自我的小Q无心刷题,于是怂恿小C和他一起颓废,玩起了一款战略游戏. 这款战略游戏的地图由n个城市以及m条连接这些城市的双向道路构成,并且从任意一个城市出发总能沿着 ...
- 仙人掌&圆方树
仙人掌&圆方树 Tags:图论 [x] [luogu4320]道路相遇 https://www.luogu.org/problemnew/show/P4320 [ ] [SDOI2018]战略 ...
- UOJ.87.mx的仙人掌(圆方树 虚树)(未AC)
题目链接 本代码10分(感觉速度还行..). 建圆方树,预处理一些东西.对询问建虚树. 对于虚树上的圆点直接做:对于方点特判,枚举其所有儿子,如果子节点不在该方点代表的环中,跳到那个点并更新其val, ...
随机推荐
- Maven下的SpringMVC MyBatis
从头开始采用Maven管理,Spring.MyBatis.Tomcat. 在配置过程中SQL Server的Jar老是加载不了,解决方案参考前一篇博文. eclipse中已经自带了Maven的插件所以 ...
- dig挖出DNS的秘密
[最简单的dig用法] 最简单的dig用法,当然就是直接输入dig按回车. 1 2 3 4 5 6 $ dig ; <<>> DiG 9.8.2rc1-RedHat-9.8 ...
- JetBrains Rider 破解 (ideaIU等等开发工具都通用)2018-02-27
贴一下Rider下载地址:(下载不了可以用百度云离线下载) Win:https://download.jetbrains.com/resharper/JetBrains.Rider-2017.3.1. ...
- 【Unity3D技术文档翻译】第1.7篇 AssetBundles 补丁更新
上一章:[Unity3D技术文档翻译]第1.6篇 使用 AssetBundle Manager 本章原文所在章节:[Unity Manual]→[Working in Unity]→[Advanced ...
- 通过实例介绍Android App自动化测试框架--Unittest
1.为什么需要使用框架实现自动化测试 作为测试工程师,可能在代码能力上相比开发工程师要弱一点,所以我们在写脚本的时候就会相对容易的碰到更多的问题,如果有一个成熟的框架供给我们使用的话,可以帮助我们避免 ...
- [翻译] 编写高性能 .NET 代码--第二章 GC -- 减少分配率, 最重要的规则,缩短对象的生命周期,减少对象层次的深度,减少对象之间的引用,避免钉住对象(Pinning)
减少分配率 这个几乎不用解释,减少了内存的使用量,自然就减少GC回收时的压力,同时降低了内存碎片与CPU的使用量.你可以用一些方法来达到这一目的,但它可能会与其它设计相冲突. 你需要在设计对象时仔细检 ...
- Servlet中文乱码问题解决办法
首先对于源jsp网站和servlet里面的字符集要一样,一般支持中文的字符集为UTF-8最好采用这个字符集(除此之外还有gb2312); 对于源jsp文件的代码中需要设置 设置你的page里面的字符集 ...
- yaf框架加载全局公共函数
在Boostrap里面建一个方法(按规则命名的函数都会被自动执行) public function _initCommonFunctions(){ Yaf_Loader::import(Yaf_App ...
- 《设计模式之禅》--设计模式大PK
创建类模式包括工厂方法模式.建造者模式.抽象工厂模式.单例模式和原型模式. 其中单例模式要保持在内存中只有一个对象,原型模式是要求通过复制的方式产生一个新的对象. [工厂方法(抽象工厂) VS 建造者 ...
- UESTC - 1057 秋实大哥与花 线段树
题意 秋实大哥是一个儒雅之人,昼听笙歌夜醉眠,若非月下即花前. 所以秋实大哥精心照料了很多花朵.现在所有的花朵排成了一行,每朵花有一个愉悦值. 秋实大哥每天要对着某一段连续的花朵歌唱,然后这些花朵的愉 ...