算法学习-最小生成树

1. Prim算法求最小生成树

题目:

给定一个 n 个点 m条边的无向图,图中可能存在重边和自环,边权可能为负数。

求最小生成树的树边权重之和,如果最小生成树不存在则输出 impossible

给定一张边带权的无向图 G=(V,E),其中 V 表示图中点的集合,E 表示图中边的集合,n=|V|,m=|E|。

由 V 中的全部 n 个顶点和 E 中 n−1 条边构成的无向连通子图被称为 G 的一棵生成树,其中边的权值之和最小的生成树被称为无向图 G的最小生成树。

输入格式

第一行包含两个整数 n 和 m。

接下来 m 行,每行包含三个整数 u,v,w,表示点 u 和点 v 之间存在一条权值为 w的边。

输出格式

共一行,若存在最小生成树,则输出一个整数,表示最小生成树的树边权重之和,如果最小生成树不存在则输出 impossible

数据范围

1≤n≤500,1≤m≤10^5,
图中涉及边的边权的绝对值均不超过 10000。

输入样例:
4 5
1 2 1
1 3 2
1 4 3
2 3 2
3 4 4
输出样例:
6

 代码:

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
typedef pair PII;
const int N = 510,INF = 0x3f3f3f3f;
int n,m;
int g[N][N];
int dist[N];
bool st[N];
int prim(){
	memset(dist,INF,sizeof dist);
	int res = 0;
	for(int i = 0;i < n;i ++){
		int t = -1;
		for(int j = 1;j <= n;j ++)
			if(!st[j] && (t == -1 || dist[t] > dist[j]))
				t = j;
		if(i && dist[t] == INF) 
			return INF;
		if(i) res += dist[t];
		for(int j = 1;j <= n;j ++)
			dist[j] = min(dist[j],g[t][j]);
		st[t] = true;
	}
	return res;
}
int main(){
	memset(g,INF,sizeof g);
	cin >> n >> m;
	while(m --){
		int a,b,c;
		cin >> a >> b >> c;
		g[a][b] = g[b][a] = min(g[a][b],c);
	}
	int t = prim();
	if(t == INF)
		cout << "impossible" << endl;
	else cout << t << endl;
	return 0;
}

2. Kruskal算法求最小生成树 

题目:

给定一个 n 个点 m条边的无向图,图中可能存在重边和自环,边权可能为负数。

求最小生成树的树边权重之和,如果最小生成树不存在则输出 impossible

给定一张边带权的无向图 G=(V,E),其中 V 表示图中点的集合,E 表示图中边的集合,n=|V|,m=|E|。

由 V 中的全部 n 个顶点和 E 中 n−1 条边构成的无向连通子图被称为 G 的一棵生成树,其中边的权值之和最小的生成树被称为无向图 G的最小生成树。

输入格式

第一行包含两个整数 n 和 m。

接下来 m 行,每行包含三个整数 u,v,w,表示点 u 和点 v 之间存在一条权值为 w的边。

输出格式

共一行,若存在最小生成树,则输出一个整数,表示最小生成树的树边权重之和,如果最小生成树不存在则输出 impossible

数据范围

1≤n≤10^5,1≤m≤2∗10^5,
图中涉及边的边权的绝对值均不超过 1000。

输入样例:
4 5
1 2 1
1 3 2
1 4 3
2 3 2
3 4 4
输出样例:
6

代码:

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
typedef pair PII;
const int N = 2e5 + 10;
int n,m,p[N];
struct edge{
	int u,v,w;
	bool operator< (const edge &W) const{
		return w < W.w;
	}
}edges[N];
int find(int x){
	if(p[x] != x)
		p[x] = find(p[x]);
	return p[x];
}
void merge(int x,int y){
	p[find(x)] = p[y];
}
int main(){
	cin >> n >> m;
	for(int i = 1;i <= n;i ++)
		p[i] = i;
	for(int i = 0;i < m;i ++){
		int a , b , c;
		cin >> a >> b >> c;
		edges[i] = {a,b,c};
	}
	sort(edges,edges + m);
	int res = 0,cnt = 1;
	for(int i = 0;i < m;i ++){
		if(find(edges[i].u) != find(edges[i].v)){
			merge(edges[i].u,edges[i].v);
			cnt ++;
			res += edges[i].w;
		}
	}
	//cout << cnt << "  " << res << endl;
	if(cnt < n) cout << "impossible" << endl;
	else cout << res << endl;
	return 0;
}

 

你可能感兴趣的:(算法,算法,学习,图论)