BZOJ3669 [Noi2014]魔法森林 LCT

欢迎访问~原文出处——博客园-zhouzhendong

去博客园看该题解


题目传送门 - BZOJ3669


题意概括

  有一个无向图,每条边分别有a、b两种权值。

  你要通过他,那么你自身的a、b两种权值必须得都不小于该边。

  现在你要从1走到n,问你自身的a+b最小为多少。


 

题解

  我们可以按照a排序。

  然后依次加边。

  那么当前最大的a就是当前加入边的a。

  至于b,我们可以写LCT来维护。

  我们在加入一条边的时候,要判断当前的边连接的两个点是否连通。

  如果不连通,那么直接加入此边。

  如果连通,那么要找出两点之间的树链的最大边权b,然后在当前边和那条边中选择b小的留下。

  每次更新答案,只需要判断1和n是否连通然后求一下值就可以了,应该是很简单的。

  至于听的一头雾水的同学们,如何维护边??那么,建议你看看这个↓

  BZOJ2594 [Wc2006]水管局长数据加强版 LCT kruskal


 

代码

#include <cstring>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <cstdlib>
using namespace std;
const int N=50005,M=100005,S=N+M;
int n,m;
int fa[S],son[S][2],rev[S],Max[S],val[S];
struct Edge{
	int x,y,a,b;
	void read(){
		scanf("%d%d%d%d",&x,&y,&a,&b);
	}
}e[M];
void clear(int x,int v){
	fa[x]=son[x][0]=son[x][1]=rev[x]=0;
	Max[x]=val[x]=v;
}
bool isroot(int x){
	return son[fa[x]][1]!=x&&son[fa[x]][0]!=x;
}
void pushup(int x){
	int ls=son[x][0],rs=son[x][1];
	Max[x]=e[Max[ls]].b>e[Max[rs]].b?Max[ls]:Max[rs];
	Max[x]=e[val[x]].b>e[Max[x]].b?val[x]:Max[x];
}
void pushdown(int x){
	if (rev[x]){
		rev[x]=0;
		rev[son[x][0]]^=1;
		rev[son[x][1]]^=1;
		swap(son[x][0],son[x][1]);
	}
}
void pushadd(int x){
	if (!isroot(x))
		pushadd(fa[x]);
	pushdown(x);
}
int wson(int x){
	return son[fa[x]][1]==x;
}
void rotate(int x){
	if (isroot(x))
		return;
	int y=fa[x],z=fa[y],L=wson(x),R=L^1;
	if (!isroot(y))
		son[z][wson(y)]=x;
	fa[x]=z,fa[y]=x,fa[son[x][R]]=y;
	son[y][L]=son[x][R],son[x][R]=y;
	pushup(y),pushup(x);
}
void splay(int x){
	pushadd(x);
	for (int y=fa[x];!isroot(x);rotate(x),y=fa[x])
		if (!isroot(y))
			rotate(wson(x)==wson(y)?y:x);
}
void access(int x){
	for (int t=0;x;t=x,x=fa[x]){
		splay(x);
		son[x][1]=t;
		pushup(x);
	}
}
void rever(int x){
	access(x);
	splay(x);
	rev[x]^=1;
}
void link(int x,int y){
	rever(x);
	fa[x]=y;
}
void split(int x,int y){
	rever(y);
	access(x);
	splay(x);
}
void cut(int x,int y){
	split(x,y);
	fa[y]=son[x][0]=0;
}
int find(int x){
	access(x);
	splay(x);
	while (son[x][0])
		x=son[x][0];
	return x;
}
bool cmp(Edge a,Edge b){
	return a.a<b.a;
}
int main(){
	scanf("%d%d",&n,&m);
	for (int i=1;i<=m;i++)
		e[i].read();
	sort(e+1,e+m+1,cmp);
	e[0].b=0;
	for (int i=1;i<=n+m;i++)
		clear(i,0);
	int ans=2333333;
	for (int i=1;i<=m;i++){
		int a=e[i].a,b=e[i].b,x=e[i].x,y=e[i].y;
		if (x==y)
			continue;
		bool flag=1;
		if (find(x)==find(y)){
			split(x,y);
			int ce=Max[x];
			if (e[ce].b>b){
				cut(e[ce].x,ce+n);
				cut(e[ce].y,ce+n);
			}
			else 
				flag=0;
		}
		if (flag){
			clear(i+n,i);
			link(x,i+n);
			link(y,i+n);
		}
		if (find(1)==find(n)){
			split(1,n);
			int ne=Max[1];
			ans=min(ans,a+e[ne].b);
		}
	}
	printf("%d",ans==2333333?-1:ans);
	return 0;
}

  

posted @ 2017-12-15 13:47  zzd233  阅读(269)  评论(0编辑  收藏  举报