【题解】CF487E Tourists

CF487E Tourists

写完这题已经完全自闭了 调了好久……

题目大意

就是求一张图中两点间所有路径中经过的点的最小值,带修。

解法

我们先考虑一下性质:对于无向图显然不好做,考虑一下咋转化成一棵树。

那就往圆方树考虑呗,本题有啥性质?

观察到:

对于一个点双,必然存在一条路径走过该点双中的最小值。

证明比较简单。由于点双内两点之间没有一个必经点,所以我们必定可以留出一个点从走过的那个最小值点再走出去。

那么就考虑建立圆方树叭。码套路,先上一个 tarjan 找点双……

细节1 注意点双的形成是需要满足 low[j]>=dfn[x] 的,不要犯傻打错了。

好的,接下来怎么修改?考虑回答询问的同时其实也就能想到怎么维护修改:扔一个树剖就好了。

那就考虑写树剖叭。码套路,先上两个 dfs ……

细节2 注意树剖的 query 啥的别打错了……

好嘞,上述准备工作都做好了,写个修改测样例……挂掉了。

冷静分析:发现不论改方点还是圆点,都会影响到周围的点。

怎么办,暴力修改?那显然不切实际,一个菊花图就可以卡爆你。

继续考虑如何高效维护。我们知道,一个方点代表一个点双,那么,对于圆方树的树结构,每一个方点的答案其实除了它的孩子还要包含其父亲。这点也就恰好让维护变得不方便起来。

那么我们不妨先考虑无视掉父亲的影响,只维护孩子,怎么做?

维护一个 multiset 对每个方点,每次直接拿这个来暴力更新方点权值。

好的,现在这里可以实现了,那带上父亲呢?

冷静分析一波,这个父亲的影响需要满足两个条件:

  • 这个点是方点

  • 这个点的父亲没有被计算到

那同时满足下列两个条件的点就只有可能是……

最近公共祖先 \((LCA)\)

所以对于每一组询问实际只有一个点需要被特殊对待!

那么就直接做就好了。

最后再附赠细节3:

更新 low[x] 不要拿 low[j] 啊……(在不 dfs 的时候),调了好久……

#include<bits/stdc++.h>
using namespace std;
#define int long long
const int N=200100;
int head[N],tot,Head[N],tto,n,m,q,w[N];
struct E {
	int nxt,to;
} e[N<<1],edge[N<<1];
inline int read(){
	int s=0;
	char ch=getchar();
	while(!isdigit(ch))ch=getchar();
	while(isdigit(ch)){
		s=s*10-'0'+ch;
		ch=getchar();
	} 
	return s;
}
int nodepos,ddf,rk[N],cir[N],To[N];
inline int Min(int x,int y) {return x<y?x:y;}
multiset<int> S[N];
inline void link(int x,int y,int w=0) {
	if(w) {
		edge[++tto]=(E) {
			Head[x],y
		};
		Head[x]=tto;
		return;
	}
	e[++tot]=(E) {
		head[x],y
	};
	head[x]=tot;
}
int son[N],siz[N],dfn[N],low[N],st[N],top;
int ttop[N],pa[N],dfstime,vis[N],dep[N];
void dfs1(int x,int fa) {
	siz[x]=1;
	dep[x]=dep[fa]+1;
	pa[x]=fa;
	for(int i=Head[x]; i; i=edge[i].nxt) {
		int j=edge[i].to;
		if(j==fa)continue;
		dfs1(j,x);
		siz[x]+=siz[j];
		if(siz[j]>siz[son[x]])son[x]=j;
	}
}
void dfs2(int x,int t) {
	rk[dfn[x]=++ddf]=x;
	ttop[x]=t;
	if(!son[x])return;
	dfs2(son[x],t);
	for(int i=Head[x]; i; i=edge[i].nxt) {
		int j=edge[i].to;
		if(j==son[x]||j==pa[x])continue;
		dfs2(j,j);
	}
}
void tarjan(int x,int root) {
	dfn[x]=low[x]=++dfstime;
	st[++top]=x;
	if(x==root&&!head[x]) {
		++nodepos;
		To[nodepos]=x;
		cir[x]=nodepos;
		link(x,nodepos,1);
		link(nodepos,x,1);
		return;
	}
	int ch=0;
	for(int i=head[x]; i; i=e[i].nxt) {
		int j=e[i].to;
		if(!dfn[j]) {
			++ch;
			tarjan(j,root);
			low[x]=Min(low[x],low[j]);
			if(dfn[x]<=low[j]&&x!=root)vis[x]=1;
			if(x==root&&ch>1)vis[x]=1;
			if(low[j]>=dfn[x]) {
				int vex=-1;
				++nodepos;
				To[nodepos]=x;
				do {
					vex=st[top--];
					cir[vex]=nodepos;
					link(vex,nodepos,1);
					link(nodepos,vex,1);
				} while(vex!=j);
				cir[x]=nodepos;
				link(nodepos,x,1);
				link(x,nodepos,1);
			}
		} else low[x]=Min(low[x],dfn[j]);
	}
}
namespace SGT {
	int mi[N<<2],ls[N<<2],rs[N<<2],node;
	inline void pushup(int x) {
		mi[x]=Min(mi[ls[x]],mi[rs[x]]);
	}
	void build(int &x,int l,int r) {
		x=++node;
		if(l==r) {
			mi[x]=w[rk[l]];
			return;
		}
		int mid=(l+r)>>1;
		build(ls[x],l,mid);
		build(rs[x],mid+1,r);
		pushup(x);
	}
	void change(int x,int L,int R,int pos,int v) {
		if(L==R) {
			mi[x]=v;
			return;
		}
		int mid=(L+R)>>1;
		if(pos<=mid)change(ls[x],L,mid,pos,v);
		else change(rs[x],mid+1,R,pos,v);
		pushup(x);
	}
	int query(int x,int L,int R,int l,int r) {
		if(L>=l&&R<=r){return mi[x];} 
		int mid=(L+R)>>1,res=(1LL<<60);
		if(l<=mid)res=query(ls[x],L,mid,l,r);
		if(mid<r)res=Min(res,query(rs[x],mid+1,R,l,r));
		return res;
	}
}
using namespace SGT;
int rt;
int Query(int u,int v) {
	int res=(1LL<<60);
	while(ttop[u]!=ttop[v]) {
		if(dep[ttop[u]]<dep[ttop[v]])swap(u,v);
		res=Min(res,query(rt,1,nodepos,dfn[ttop[u]],dfn[u]));
		u=pa[ttop[u]];
	}
	if(dep[u]<dep[v])swap(u,v);
	res=Min(res,query(rt,1,nodepos,dfn[v],dfn[u]));
	if(dep[u]>dep[v])swap(u,v);
	if(u>n)res=Min(res,w[pa[u]]);
	return res;
}
void Change(int pos,int v) {
	change(rt,1,nodepos,dfn[pos],v);
	if(pos==1) {
		w[pos]=v;
		return;
	}
	int o=pa[pos];
	S[o-n].erase(S[o-n].find(w[pos]));
	S[o-n].insert(v);
	int minv=*S[o-n].begin();
	if(minv==w[o]){
		w[pos]=v;
		return;
	}
	change(rt,1,nodepos,dfn[o],minv);
	w[o]=minv;
	w[pos]=v;
}
inline void write(int x){
	if(x>9)write(x/10);
	putchar(x%10+'0');
}
signed main() {
	freopen("in.txt","r",stdin);
	n=read();m=read();q=read();
	for(int i=1; i<=n; ++i)w[i]=read();
	for(int i=1; i<=m; ++i) {
		int u=read(),v=read();
		link(u,v);
		link(v,u);
	}
	nodepos=n;
	tarjan(1,1);
	memset(dfn,0,sizeof dfn);
	dfs1(1,0);
	dfs2(1,1);
	for(int i=2; i<=n; ++i) {
		S[pa[i]-n].insert(w[i]);
	}
	for(int i=n+1; i<=nodepos; ++i)
		w[i]=(S[i-n].empty()?(1LL<<60):(*S[i-n].begin()));
	memset(mi,0x7f,sizeof mi);
	build(rt,1,nodepos);
	while(q--) {
		int u,v;
		char opt;
		cin>>opt;
		u=read();v=read();
		if(opt=='A') {
			write(Query(u,v));
			putchar('\n');
			continue;
		}
		Change(u,v);
	}
	return 0;
}
posted @ 2021-09-20 20:18  Refined_heart  阅读(44)  评论(0编辑  收藏  举报