P6175 无向图的最小环问题

无向图的最小环问题

题目描述

给定一张无向图,求图中一个至少包含 \(3\) 个点的环,环上的节点不重复,并且环上的边的长度之和最小。该问题称为无向图的最小环问题。在本题中,你需要输出最小的环的边权和。若无解,输出 No solution.

输入格式

第一行两个正整数 \(n,m\) 表示点数和边数。

接下来 \(m\) 行,每行三个正整数 \(u,v,d\),表示节点 \(u,v\) 之间有一条长度为 \(d\) 的边。

输出格式

输出边权和最小的环的边权和。若无解,输出 No solution.

样例 #1

样例输入 #1

5 7
1 4 1
1 3 300
3 1 10
1 2 16
2 3 100
2 5 15
5 3 20

样例输出 #1

61

提示

样例解释:一种可行的方案:\(1-3-5-2-1\)

对于 \(20\%\) 的数据,\(n,m \leq 10\)

对于 \(60\%\) 的数据,\(m\leq 100\)

对于 \(100\%\) 的数据,\(1\le n\leq 100\)\(1\le m\leq 5\times 10^3\)\(1 \leq d \leq 10^5\)

Floyd 求最小环

image

注意读入初始数据建边的时候的细节!!!g[][] dis[][]都要取Min 少一个都错!!!

#include<bits/stdc++.h>
using namespace std;
#define int long long
const int N=205;
const int INF=1e9;
int n,m,g[N][N],dis[N][N];
signed main()
{
	ios::sync_with_stdio(false);
	cin>>n>>m;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++)
			if(i!=j)dis[i][j]=g[i][j]=INF;
	for(int i=1;i<=m;i++)
	{
		int u,v,w;
		cin>>u>>v>>w;//可能有重边 所以我们要取最短的边 
		dis[u][v]=min(dis[u][v],w);
		dis[v][u]=min(dis[v][u],w);
		g[u][v]=min(g[u][v],w);
		g[v][u]=min(g[v][u],w);
	}
	int minn=INF;
	for(int k=1;k<=n;k++)
	{//注意 求环的时候 i j 是要 < k 的 
		for(int i=1;i<k;i++)
			for(int j=i+1;j<k;j++)
				minn=min(minn,dis[i][j]+g[j][k]+g[k][i]);
	// Floyd 更新dis[][]时不用规定大小 
		for(int i=1;i<=n;i++)
			for(int j=1;j<=n;j++)
				if(i!=j&&j!=k)
					dis[i][j]=min(dis[i][j],dis[i][k]+dis[k][j]),dis[j][i]=dis[i][j];
	}
	if(minn==INF)cout<<"No solution.\n";
	else cout<<minn<<"\n";
	return 0;
}
posted @ 2023-04-13 20:06  N0zoM1z0  阅读(15)  评论(0编辑  收藏  举报