POJ2395 最小生成树裸题三种复杂度做法

//KRUSCAL
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
const int maxn=2010,maxm=10010;
struct node{
	int x,y,w;
};
struct node a[maxm],tmp;
int f[maxn],rank[maxn];
void qsort(int l,int r){
	int mid=a[(l+r)/2].w,i,j;
	i=l;j=r;
	while(i<j){
		while(i<=j && a[i].w<mid)i++;
		while(i<=j && a[j].w>mid)j--;
		if(i<=j){
			tmp=a[i];a[i]=a[j];a[j]=tmp;
			i++;j--;
		}
	}
	if(l<j)qsort(l,j);
	if(i<r)qsort(i,r);
}
int fa(int u){
	if(f[u]!=u)
		f[u]=fa(f[u]);
	return f[u];
}
int find(int u,int v){
	return fa(u)==fa(v)?0:1;
}
void link(int u,int v){
	if(rank[fa(u)]>=rank[fa(v)]){
		f[fa(v)]=fa(u);
		rank[fa(u)]++;
	}else{
		f[fa(u)]=fa(v);
		rank[fa(v)]++;
	}
}
int main(){
#ifndef ONLINE_JUDGE
	freopen("prim.in","r",stdin);
//	freopen("prim.out","w",stdout);
#endif
	int i,j,k,m,n,s;
	scanf("%d%d",&n,&m);
	for(i=1;i<=n;i++)f[i]=i;
	for(i=1;i<=m;i++)
		scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].w);
	qsort(1,m);
	s=n-1;i=0;
	int ans=0;
	while(s && i<=m){
		i++;
		if(find(a[i].x,a[i].y)){
			link(a[i].x,a[i].y);
			s--;
			ans=ans>a[i].w?ans:a[i].w;			
		}		
	}
	printf("%d\n",ans);
	return 0;
}


PRIM(N^2)

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
const int maxn=2010,maxm=10010,INF=1000001000;
int a[maxn][maxn],d[maxn],p[maxn];
int main(){
#ifndef ONLINE_JUDGE
	freopen("prim.in","r",stdin);
	//freopen("prim.out","w",stdout);
#endif
	int i,j,k,m,n,s;
	int x,y,w,ans=0,min;
	scanf("%d%d",&n,&m);
	for(i=1;i<=n;i++)d[i]=INF;
	for(i=1;i<=n;i++)
		for(j=1;j<=n;j++)
				a[i][j]=INF;
	for(i=1;i<=m;i++){
		scanf("%d%d%d",&x,&y,&w);
		if(a[x][y]>w)a[x][y]=a[y][x]=w;
	}
	d[1]=0;
	for(i=1;i<=n;i++){
		min=INF;
		for(j=1;j<=n;j++)
			if(!p[j] && d[j]<min){
				min=d[j];
				k=j;
			}
		p[k]=1;		
		ans=ans>d[k]?ans:d[k];
		for(j=1;j<=n;j++)
			if(a[k][j] && d[j]>a[k][j])
				d[j]=a[k][j];
	}
	printf("%d\n",ans);
	return 0;
}

PRIM NLOG2N

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
const int Maxn=2010,Maxm=50010,Inf=1000010000;
int e,to[Maxm],ws[Maxm],begin[Maxn],next[Maxm];
int d[Maxn],p[Maxn],q[Maxn],pos[Maxn];
void add(int x,int y,int z){
	to[++e]=y;
	next[e]=begin[x];
	begin[x]=e;
	ws[e]=z;
}
int main(){
	//freopen("prim.in","r",stdin);
	//freopen("prim.out","w",stdout);
	int i,j,k,m,n,l;
	int x,y,z,u,v,w;
	scanf("%d%d",&n,&m);
	for(i=1;i<=m;i++){
		scanf("%d%d%d",&x,&y,&z);
		add(x,y,z);
		add(y,x,z);				
	}
	for(i=1;i<=n;i++)d[i]=Inf;
	d[1]=0;l=n;
	for(i=1;i<=n;i++){q[i]=i;pos[i]=i;}
	int ans=0;
	for(i=1;i<=n;i++){
		k=q[1];
		q[1]=q[l];pos[q[l]]=1;l--;
		u=1;				
		while( (u*2<=l && d[q[u]]>d[q[2*u]]) || (u*2+1<=l && d[q[u]]>d[q[2*u+1]]) ){
			v=u*2;
			if(v+1<=l && d[q[v]]>d[q[v+1]])v++;
			w=q[u];q[u]=q[v];q[v]=w;
			w=pos[q[u]];pos[q[u]]=pos[q[v]];pos[q[v]]=w;
			u=v;			
		}
		p[k]=1;
		ans=ans>d[k]?ans:d[k];
		for(j=begin[k];j;j=next[j]){
			u=to[j];
			if(!p[u] && d[u]>ws[j])d[u]=ws[j];
			v=pos[u];
			while(v>=1 && d[q[v]]<d[q[v/2]]){
				w=q[v];q[v]=q[v/2];q[v/2]=w;
				w=pos[q[v]];pos[q[v]]=pos[q[v/2]];pos[q[v/2]]=w;				
				v=v/2;
			}			
		}
	}
	printf("%d\n",ans);	
	return 0;
}


你可能感兴趣的:(POJ2395 最小生成树裸题三种复杂度做法)