原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ30.html

题目传送门 - UOJ#30

题意

  uoj写的很简洁、清晰,这里就不抄一遍了。

题解

  首先建出圆方树。接下来,我们称"圆点"为原来有的点,"方点"为新增的点。

  然后先只考虑在线询问如何做。

    ——把方点的值设置成所有与他连边的圆点的权值的最小值,直接在圆方树上树链剖分再套个线段树支持一下区间询问即可。

  然后会发现这样做支持不了修改操作。

    ——直接来个菊花图不断修改根节点就GG了。

  于是我们考虑进一步想办法。

  我们把方点的值重新定义成“在圆方树上,该点的儿子的权值的最小值”。那么,在询问的时候,其他都一样,但是如果 lca 为方点,那么加上其 fa 对 min 的贡献即可。

  时间复杂度 $O(n\log ^2 n )$ 。

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
LL read(){
LL x=0,f=1;
char ch=getchar();
while (!isdigit(ch)&&ch!='-')
ch=getchar();
if (ch=='-')
f=-1,ch=getchar();
while (isdigit(ch))
x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return x*f;
}
const int N=200005,INF=1.1e9;
int pr[N],dfn[N],low[N],st[N],Time,top,tot;
multiset <int> Mins[N];
vector <int> G[N],g[N];
void Tarjan(int x){
low[x]=dfn[x]=++Time,st[++top]=x;
for (auto y : G[x])
if (!dfn[y]){
Tarjan(y);
low[x]=min(low[x],low[y]);
if (low[y]>=dfn[x]){
tot++;
g[x].push_back(tot);
g[tot].push_back(x);
int z;
do {
z=st[top--];
g[z].push_back(tot);
g[tot].push_back(z);
} while (z!=y);
}
}
else
low[x]=min(low[x],dfn[y]);
}
namespace sp{
int n,outn;
int fa[N],son[N],size[N],depth[N],top[N],p[N],ap[N],cnp=0,val[N];
void dfs(int x,int pre,int d){
size[x]=1,fa[x]=pre,son[x]=-1,depth[x]=d;
for (auto y : g[x])
if (y!=pre){
dfs(y,x,d+1);
size[x]+=size[y];
if (son[x]==-1||size[y]>size[son[x]])
son[x]=y;
}
}
void Get_Top(int x,int tp){
top[x]=tp;
ap[p[x]=++cnp]=x;
if (son[x]==-1)
return;
Get_Top(son[x],tp);
for (auto y : g[x])
if (y!=fa[x]&&y!=son[x])
Get_Top(y,y);
}
int Min[N<<2];
void build(int rt,int L,int R){
if (L==R)
return (void)(Min[rt]=val[ap[L]]);
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
build(ls,L,mid);
build(rs,mid+1,R);
Min[rt]=min(Min[ls],Min[rs]);
}
int query(int rt,int L,int R,int xL,int xR){
if (xL>xR||L>xR||R<xL)
return INF;
if (xL<=L&&R<=xR)
return Min[rt];
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
return min(query(ls,L,mid,xL,xR),query(rs,mid+1,R,xL,xR));
}
void update(int rt,int L,int R,int x,int d){
if (L==R)
return (void)(Min[rt]=d);
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
if (x<=mid)
update(ls,L,mid,x,d);
else
update(rs,mid+1,R,x,d);
Min[rt]=min(Min[ls],Min[rs]);
}
int query(int a,int b){
int f1=top[a],f2=top[b],ans=INF;
while (f1!=f2){
if (depth[f1]<depth[f2])
swap(f1,f2),swap(a,b);
ans=min(ans,query(1,1,n,p[f1],p[a]));
a=fa[f1],f1=top[a];
}
if (depth[a]>depth[b])
swap(a,b);
if (a>outn&&fa[a]!=0)
ans=min(ans,query(1,1,n,p[fa[a]],p[fa[a]]));
return min(ans,query(1,1,n,p[a],p[b]));
}
}
int main(){
int n=tot=read(),m=read(),q=read();
for (int i=1;i<=n;i++)
pr[i]=read();
for (int i=1;i<=m;i++){
int a=read(),b=read();
G[a].push_back(b);
G[b].push_back(a);
}
Tarjan(1);
sp :: dfs(1,0,0);
sp :: Get_Top(1,1);
for (int i=n+1;i<=tot;i++)
Mins[i].insert(INF);
for (int i=1;i<=n;i++)
Mins[sp :: fa[i]].insert(pr[i]);
for (int i=1;i<=n;i++)
sp :: val[i]=pr[i];
for (int i=n+1;i<=tot;i++)
sp :: val[i]=*Mins[i].begin();
sp :: build(1,1,sp :: n=tot);
sp :: outn=n;
while (q--){
char s[10];
scanf("%s",s);
int x=read(),y=read();
if (s[0]=='A')
printf("%d\n",sp :: query(x,y));
else {
int f=sp :: fa[x];
if (f){
Mins[f].erase(Mins[f].find(pr[x]));
Mins[f].insert(pr[x]=y);
sp :: update(1,1,sp :: n,sp :: p[f],*Mins[f].begin());
}
sp :: update(1,1,sp :: n,sp :: p[x],y);
}
}
return 0;
}

  

UOJ#30/Codeforces 487E Tourists 点双连通分量,Tarjan,圆方树,树链剖分,线段树的更多相关文章

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

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

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

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

  3. Educational Codeforces Round 3 E. Minimum spanning tree for each edge 最小生成树+树链剖分+线段树

    E. Minimum spanning tree for each edge time limit per test 2 seconds memory limit per test 256 megab ...

  4. Codeforces Round #200 (Div. 1) D. Water Tree 树链剖分+线段树

    D. Water Tree time limit per test 4 seconds memory limit per test 256 megabytes input standard input ...

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

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

  6. CF487 E. Tourists [点双连通分量 树链剖分 割点]

    E. Tourists 题意: 无向连通图 C a w: 表示 a 城市的纪念品售价变成 w. A a b: 表示有一个游客要从 a 城市到 b 城市,你要回答在所有他的旅行路径中最低售价的最低可能值 ...

  7. Simple Cycles Edges CodeForces - 962F(点双连通分量)

    题意: 求出简单环的所有边,简单环即为边在一个环内 解析: 求出点双连通分量,如果一个连通分量的点数和边数相等,则为一个简单环 点双连通分量  任意两个点都至少存在两条点不重复的路径  即任意两条边都 ...

  8. codeforces 487E Tourists

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

  9. 点/边 双连通分量---Tarjan算法

    运用Tarjan算法,求解图的点/边双连通分量. 1.点双连通分量[块] 割点可以存在多个块中,每个块包含当前节点u,分量以边的形式输出比较有意义. typedef struct{ //栈结点结构 保 ...

随机推荐

  1. ecshop二次开发 给商品添加自定义字段【包含我自己进一步的开发实例详解】

    本文包含商品自定义添加教程及进一步的开发实例: 教程: 说起自定义字段,我想很多的朋友像我一样会想起一些开源的CMS(比如Dedecms.Phpcms.帝国)等,他们是可以在后台直接添加自定义字段的. ...

  2. 浅析python 中__name__ = &#39;__main__&#39; 的作用

    引用http://www.jb51.net/article/51892.htm 很多新手刚开始学习python的时候经常会看到python 中__name__ = \'__main__\' 这样的代码 ...

  3. COJ980 WZJ的数据结构(负二十)

    试题描述 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走,现在有Q组询问,每组询问询问从点v开始只 ...

  4. 简单版解决IE兼容性问题

    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1" /> 在访问页面时, 如 ...

  5. Windows7 64位环境6sv2.1大气传输模型修改源码添加国产高分卫星GF-1 GF-2光谱响应支持

    下面开始添加国产卫星光谱响应的支持: 以下主要参考文章“6S大气传输模型修改源码添加.自定义卫星光谱响应(以HJ-1B CCD为例)”网址:http://blog.csdn.net/sam92/art ...

  6. C#读写Excel实践笔记

    使用第三类包:NPOI 介绍 Github地址:https://github.com/tonyqus/npoi,Java POI项目的.NET版. 通过它可以在没有安装Office软件的情况下,快速的 ...

  7. JVM内核优化

    1.垃圾回收器 JVM垃圾回收器有串行和并行两种. 1.1 Serial收集器(串行,单线程),现在使用较少 Serial一般收集新生代 SerialOld一般收集老年代(采用标记压缩算法) 1.2 ...

  8. JAVA面向对象编程之购物车界面的设计与功能的实现

    1.小组成员及分工 小组成员 负责工作 兰泽祥 负责总体界面.Git代码库 吴修恩 负责MVC模式.类名包名的规范化 2.Git 仓库地址 仓库地址:https://gitee.com/lanzexi ...

  9. 使用Vagrant和VirtualBox一步步地创建一个Base Box

    box集合  http://www.vagrantbox.es/ Vagrant和VirtualBox软件的安装步骤省略,去官网下载最新的版本然后下一步下一步地安装就行了,和正常的安装软件没有什么区别 ...

  10. MVC :“已添加了具有相同键的项”

    最近将一个项目从ASP.NET MVC 3升级至刚刚发布的ASP.NET MVC 5.1,升级后发现一个ajax请求出现了500错误,日志中记录的详细异常信息如下: System.ArgumentEx ...