题意

​ \(n\) 个点 \(m\) 条边的无向连通图,每个点有点权,\(q\) 个要求,每次更新一个点的点权或查询两点间路径权值最小的点最小的路径。

思路

​ 算是圆方树的板子吧?圆方树处理的主要就是两点之间路径的问题。

​ 我们先对原图建一棵圆方树,然后每个圆点的信息传递给父亲,一定是方点,用堆维护信息。最后只要树剖线段树查路径最小值即可,注意特判 \(\rm{lca}\) 即可。

代码

#include<bits/stdc++.h>
#define FOR(i, x, y) for(int i = (x), i##END = (y); i <= i##END; ++i)
#define DOR(i, x, y) for(int i = (x), i##END = (y); i >= i##END; --i)
template<typename T, typename _T> inline bool chk_min(T &x, const _T &y) {return y < x ? x = y, 1 : 0;}
template<typename T, typename _T> inline bool chk_max(T &x, const _T &y) {return x < y ? x = y, 1 : 0;}
typedef long long ll;
const int N = 100005;
const int M = 100005; template<const int N, const int M, typename T> struct Linked_List
{
int head[N], nxt[M], tot; T to[M];
Linked_List() {clear();}
T &operator [](const int x) {return to[x];}
void clear() {memset(head, -1, sizeof(head)), tot = 0;}
void add(int u, T v) {to[tot] = v, nxt[tot] = head[u], head[u] = tot++;}
#define EOR(i, G, u) for(int i = G.head[u]; ~i; i = G.nxt[i])
}; struct Segment_Tree
{
int mi[N << 3];
void update(int k, int x, int val, int l, int r)
{
if(l == r) {mi[k] = val; return;}
int mid = (l + r) >> 1;
if(x <= mid) update(k << 1, x, val, l, mid);
else update(k << 1 | 1, x, val, mid + 1, r);
mi[k] = std::min(mi[k << 1], mi[k << 1 | 1]);
}
int query(int k, int L, int R, int l, int r)
{
if(L <= l && r <= R) return mi[k];
int mid = (l + r) >> 1;
if(R <= mid) return query(k << 1, L, R, l, mid);
else if(L > mid) return query(k << 1 | 1, L, R, mid + 1, r);
else return std::min(query(k << 1, L, R, l, mid), query(k << 1 | 1, L, R, mid + 1, r));
}
}; Linked_List<N, M << 1, int> G;
Linked_List<N << 1, N << 2, int> T;
Segment_Tree ST; int dfn[N], low[N], stk[N], idxer, bcc, tp; int fa[N << 1], dep[N << 1], sz[N << 1], son[N << 1], top[N << 1];
int lfn[N << 1], rfn[N << 1], ori[N << 1], dfn_idx;
int pw[N << 1];
std::multiset<int> st[N << 1]; int n, m, q; void tarjan(int u, int fa_e)
{
dfn[u] = low[u] = ++idxer;
stk[++tp] = u; EOR(i, G, u)
{
if(i == (fa_e ^ 1)) continue;
int v = G[i];
if(!dfn[v])
{
tarjan(v, i), chk_min(low[u], low[v]);
if(low[v] >= dfn[u])
{
bcc++;
do
{
T.add(n + bcc, stk[tp]);
T.add(stk[tp], n + bcc);
}
while(stk[tp--] != v);
T.add(n + bcc, u), T.add(u, n + bcc);
}
}
else if(dfn[v] < dfn[u])
chk_min(low[u], dfn[v]);
}
} void dfs(int u, int f, int d)
{
fa[u] = f, dep[u] = d, sz[u] = 1, son[u] = 0;
EOR(i, T, u)
{
int v = T[i];
if(v == f) continue;
dfs(v, u, d + 1);
sz[u] += sz[v];
if(sz[v] > sz[son[u]]) son[u] = v;
}
} void hld(int u, int tp)
{
ori[lfn[u] = ++dfn_idx] = u;
top[u] = tp;
if(son[u]) hld(son[u], tp);
EOR(i, T, u)
{
int v = T[i];
if(v == fa[u] || v == son[u]) continue;
hld(v, v);
}
rfn[u] = dfn_idx;
} int get_lca(int u, int v)
{
while(top[u] != top[v])
{
if(dep[top[u]] < dep[top[v]]) std::swap(u, v);
u = fa[top[u]];
}
return dep[u] < dep[v] ? u : v;
} int query(int u, int v)
{
int res = 2e9;
while(top[u] != top[v])
{
if(dep[top[u]] < dep[top[v]]) std::swap(u, v);
chk_min(res, ST.query(1, lfn[top[u]], lfn[u], 1, n + bcc));
u = fa[top[u]];
}
if(dep[u] > dep[v]) std::swap(u, v);
chk_min(res, ST.query(1, lfn[u], lfn[v], 1, n + bcc));
if(u > n) chk_min(res, pw[fa[u]]);
return res;
} int main()
{
scanf("%d%d%d", &n, &m, &q);
FOR(i, 1, n) scanf("%d", &pw[i]);
FOR(i, 1, m)
{
int u, v;
scanf("%d%d", &u, &v);
G.add(u, v), G.add(v, u);
} tarjan(1, -1);
dfs(1, 0, 1), hld(1, 1); FOR(u, n + 1, n + bcc)
{
EOR(i, T, u)
{
int v = T[i];
if(v == fa[u]) continue;
st[u].insert(pw[v]);
}
pw[u] = (*st[u].begin());
} FOR(i, 1, n + bcc) ST.update(1, lfn[i], pw[i], 1, n + bcc); while(q--)
{
char str[3]; int a, b;
scanf("%s%d%d", str, &a, &b);
if(str[0] == 'C')
{
if(fa[a])
{
st[fa[a]].erase(st[fa[a]].find(pw[a]));
st[fa[a]].insert(b);
pw[fa[a]] = (*st[fa[a]].begin());
ST.update(1, lfn[fa[a]], pw[fa[a]], 1, n + bcc);
}
pw[a] = b;
ST.update(1, lfn[a], b, 1, n + bcc); }
else if(str[0] == 'A')
printf("%d\n", query(a, b));
} return 0;
}

CodeForces 487E Tourists(圆方树+线段树+树链剖分)的更多相关文章

  1. CF487E Tourists(圆方树+树链剖分+multiset/可删堆)

    CF487E Tourists(圆方树+树链剖分+multiset/可删堆) Luogu 给出一个带点权的无向图,两种操作: 1.修改某点点权. 2.询问x到y之间简单路径能走过的点的最小点权. 题解 ...

  2. Codeforces 487E Tourists [广义圆方树,树链剖分,线段树]

    洛谷 Codeforces 思路 首先要莫名其妙地想到圆方树. 建起圆方树后,令方点的权值是双联通分量中的最小值,那么\((u,v)\)的答案就是路径\((u,v)\)上的最小值. 然而这题还有修改, ...

  3. UOJ#30/Codeforces 487E Tourists 点双连通分量,Tarjan,圆方树,树链剖分,线段树

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ30.html 题目传送门 - UOJ#30 题意 uoj写的很简洁.清晰,这里就不抄一遍了. 题解 首先建 ...

  4. CF487E Tourists[圆方树+树剖(线段树套set)]

    做这题的时候有点怂..基本已经想到正解了..结果感觉做法有点假,还是看了正解题解.. 首先提到简单路径上经过的点,就想到了一个关于点双的结论:两点间简单路径上所有可能经过的点的并等于路径上所有点所在点 ...

  5. CF487E Tourists + 圆方树学习笔记(圆方树+树剖+线段树+multiset)

    QWQ果然我已经什么都学不会的人了. 这个题目要求的是图上所有路径的点权和!QWQ(我只会树上啊!) 这个如果是好啊 这时候就需要 圆方树! 首先在介绍圆方树之前,我们先来一点简单的前置知识 首先,我 ...

  6. Tourists——圆方树

    CF487E Tourists 一般图,带修求所有简单路径代价. 简单路径,不能经过同一个点两次,那么每个V-DCC出去就不能再回来了. 所以可以圆方树,然后方点维护一下V-DCC内的最小值. 那么, ...

  7. CF487E Tourists 圆方树、树链剖分

    传送门 注意到我们需要求的是两点之间所有简单路径中最小值的最小值,那么对于一个点双联通分量来说,如果要经过它,则一定会经过这个点双联通分量里权值最小的点 注意:这里不能缩边双联通分量,样例\(2\)就 ...

  8. uoj30【CF Round #278】Tourists(圆方树+树链剖分+可删除堆)

    - 学习了一波圆方树 学习了一波点分治 学习了一波可删除堆(巧用 ? STL) 传送门: Icefox_zhx 注意看代码看怎么构建圆方树的. tips:tips:tips:圆方树内存记得开两倍 CO ...

  9. codeforces 487E Tourists

    如果不是uoj上有的话(听说这是China Round),我有可能就错过这道题目了(这是我有史以来为oi写的最长的代码,用了我一天TAT!). 题目 传送门. 一个连通无向图,点上有权,支持两种操作: ...

随机推荐

  1. 关于 NuGet 本地仓库、.NET Core 引用等实战

  2. 关于 C# 8.0 的 Switch Case When 的用法

    直接贴代码了: static void Main(string[] args) { SwitchSample(); } private static void SwitchSample() { Swi ...

  3. Visual Studio 2019 (VS2019)正式版安装 VisualSVN Server 插件

    VS2019 正式版最近刚刚推出来,目前 Ankhsvn 还不支持,它最高只支持 VS2017,全网搜索了一下,也没有找到.在 Stackoverflow 上看了一下,找到这篇问答: 自己按照这种方法 ...

  4. 初探云原生应用管理之:聊聊 Tekton 项目

    [编者的话]“人间四月芳菲尽,山寺桃花始盛开.” 越来越多专门给 Kubernetes 做应用发布的工具开始缤纷呈现,帮助大家管理和发布不断增多的 Kubernetes 应用.在做技术选型的时候,我们 ...

  5. 利用 Symbol Type Viewer 工具实现将 pdb 文件 转换为 c\c++ 头文件

    利用 Symbol Type Viewer 工具实现将 pdb 文件 转换为 c\c++ 头文件 一.得到符号 二.将符号转换为 .h 文件 三.得到 c\c++ 头文件,之后编程时直接导入这个文件即 ...

  6. asp.net发布后其他电脑部署——未能加载文件或程序集 System.Web.Mvc, Version=2.0.0.0, Culture=neutral,

    本机测试及发布使用正常 在项目中添加了引用但相关dll文件未在bin文件夹中 导致发布后部署其他电脑未能加载程序集 网上说要下载相关内容在部署服务器安装 但怕在服务器安装出现其他问题 所以在项目中重新 ...

  7. Microsoft.Office.Interop.Excel 读取 excel 中的 checkbox 和 radio

    using Excel = Microsoft.Office.Interop.Excel; Excel.Application excelapp = new Excel.Application(); ...

  8. windows 下安装MongoDB

    一:下载mongodb安装包 下载地址:https://www.mongodb.com/download-center/community 这里推荐下载msi的安装包 二:安装mongodb 双击下载 ...

  9. STorM32 BGC三轴云台控制板电机驱动电路设计(驱动芯片DRV8313)

    1  序言 相信对云台有兴趣的小伙伴对STorM32 BGC这块云台控制板并不陌生,虽说这块控制板的软件已经不再开源,但是在GitHub上依旧可以找到两三个版本的代码,而硬件呢我们也可以从Olliw( ...

  10. Jquery选择器与样式操作

    jquery选择器 jquery用法思想一 选择某个网页元素,然后对它进行某种操作 jquery选择器 jquery选择器可以快速地选择元素,选择规则和css样式相同,使用length属性判断是否选择 ...