(prim算法题型一)求最小生成树的权值和、路径、边值的最小和最大值。

比较优秀的矩阵类型prim算法:

#include "stdio.h"
#include "string.h"
#define N 500
#define INT 10000
bool vis[N];
int dis[N];
int a[N][N];
int main(){
int t;
scanf("%d", &t);
while(t--){
int n;
scanf("%d", &n);
int i,j,temp,k;
memset(vis,0,sizeof(vis));
for(i=1;i<=n;++i){
for(j=1;j<=n;++j){
scanf("%d", &a[i][j]);
}
}
for(i=1;i<=n;++i){
dis[i]=INT;
}
dis[1]=0;
for(i=1;i<=n;++i){
temp=INT;
k=0;
for(j=1;j<=n;++j){
if(!vis[j]&&dis[j]<temp){
temp=dis[j];
k=j;
}
}
vis[k]=1;
for(j=1;j<=n;++j){
if(!vis[j]&&dis[j]>a[k][j]){
dis[j]=a[k][j];
}
}
}
int max=0;
for(i=1;i<=n;++i){
if(max<dis[i])
max=dis[i];
}
printf("%d\n",max);
}
return 0;
}

/*
2
3
0 990 692
990 0 179
692 179 0
*/



1.输出最小生成树个边权值累加和



4
0 4 9 21
4 0 8 17
9 8 0 16
21 17 16 0




#include <stdio.h>
#include <string.h>
#define MaxInt 0x3f3f3f3f
#define N 110
//创建map二维数组储存图表,low数组记录每2个点间最小权值,visited数组标记某点是否已访问
int map[N][N],low[N],visited[N];
int n;
 
int prim()
{
    int i,j,pos,min,result=0;
    memset(visited,0,sizeof(visited));
//从某点开始,分别标记和记录该点
    visited[1]=1;pos=1;
//第一次给low数组赋值
    for(i=1;i<=n;i++)
        if(i!=pos) low[i]=map[pos][i];
//再运行n-1次
    for(i=1;i<n;i++)
    {
//找出最小权值并记录位置
     min=MaxInt;
     for(j=1;j<=n;j++)
         if(visited[j]==0&&min>low[j])
         {
             min=low[j];pos=j;
         }
//最小权值累加
    result+=min;
//标记该点
    visited[pos]=1;
//更新权值
    for(j=1;j<=n;j++)
        if(visited[j]==0&&low[j]>map[pos][j])
            low[j]=map[pos][j];
    }
    return result;
}
 
int main()
{
    int i,v,j,ans;
    while(scanf("%d",&n)!=EOF)
    {
//所有权值初始化为最大
        memset(map,MaxInt,sizeof(map));
        for(i=1;i<=n;i++)
            for(j=1;j<=n;j++)
            {
                scanf("%d",&v);
                map[i][j]=map[i][j]=v;
            }
            ans=prim();
            printf("%d\n",ans);
    }
    return 0;
}




2、求出最小生成树中边的最大值。


1
3
0 990 692
990 0 179
692 179 0


692




#include<stdio.h>
#define MAX 505
#define inf 999999
int c[MAX][MAX];
int n;


void prim()
{
    int lowcost[MAX ];
    int closest[MAX ];
    bool s[MAX ];
    s[1]=true;
    for(int i=2;i<=n;i++)
    {
        lowcost[i]=c[1][i];
        closest[i]=1;
        s[i]=false;
    }
    for(int i=1;i<=n;i++)
    {
        int min=inf;
        int j=i;
        for(int k=2;k<=n;k++)
        if((lowcost[k]<min)&&(!s[k]))
        {
            min=lowcost[k];
            j=k;
        }






       //cout<<j<<" "<<closet[j]<<endl;输出最小生成树的路径。


        s[j]=true;
        for(int k=2;k<=n;k++)
        {
            if((c[j][k]<lowcost[k])&&(!s[k]))
            {
                lowcost[k]=c[j][k];
                closest[k]=j;
            }
        }
    }




//最小生成树的边值已经放大lowcost数组中了。遍历一下就可以得到最大最小值。
    int result=-1;
    for(int i=2;i<=n;i++)
       {
           if(result<lowcost[i])
            result=lowcost[i];
       }
       printf("%d\n",result);
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
        {
            for(int  j=1;j<=n;j++)
            {
                scanf("%d",&c[i][j]);
            }
        }
        prim();
    }
    return 0;

}











  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++ Prim算法可以用于解带权连通无向图的最小生成。下面是一个示例代码,用于输出最小生成的边及权: ```cpp #include <iostream> #include <vector> #include <climits> using namespace std; // 定义图的最大顶点数 const int MAX_V = 100; // 邻接矩阵存储图 int graph[MAX_V][MAX_V]; // Prim算法解最小生成 void primMST(int n) { // 存储最小生成的边及权 vector<pair<int, int>> mst; vector<int> key(n, INT_MAX); // 存储顶点到最小生成的最小权 vector<bool> visited(n, false); // 记录顶点是否已经加入最小生成 // 从第一个顶点开始构建最小生成 key[0] = 0; for (int i = 0; i < n - 1; i++) { // 找到当前不在最小生成中的顶点中,到最小生成的权最小的顶点 int minKey = INT_MAX, minIndex; for (int j = 0; j < n; j++) { if (!visited[j] && key[j] < minKey) { minKey = key[j]; minIndex = j; } } // 将该顶点加入最小生成 visited[minIndex] = true; // 将该顶点与最小生成中的顶点连接的边加入最小生成 for (int j = 0; j < n; j++) { if (!visited[j] && graph[minIndex][j] != 0 && graph[minIndex][j] < key[j]) { key[j] = graph[minIndex][j]; mst.push_back(make_pair(minIndex, j)); } } } // 输出最小生成的边及权之和 int sum = 0; for (auto edge : mst) { cout << edge.first + 1 << " " << edge.second + 1 << " " << graph[edge.first][edge.second] << endl; sum += graph[edge.first][edge.second]; } cout << "权之和:" << sum << endl; } int main() { int n, count; cin >> n >> count; // 初始化邻接矩阵 for (int i = 0; i < n; i++) { for (int j = 0; j < n; j++) { graph[i][j] = 0; } } // 读入三元组 for (int i = 0; i < count; i++) { int u, v, w; cin >> u >> v >> w; graph[u - 1][v - 1] = w; graph[v - 1][u - 1] = w; } // 调用Prim算法解最小生成 primMST(n); return 0; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值