C语言Prim算法和Prim-Alternat找最小生成树

1、用prim算法求最小生成树

绿色线会标记选过的边

在这里插入图片描述

从v1当作起始点开始,可选择:
(v1,v2)权值为6
(v1,v3)权值为3
(v1,v4)权值为1
从中选择边(v1,v4),最小权值为1
在这里插入图片描述

从v1和v4中选连接边,可选择的边有:
(v1,v2)权值为6
(v1,v3)权值为3
(v4,v3)权值为2
(v4,v5)权值为10
从中选择权值最小为2的边,(v4,v3)
在这里插入图片描述

v1,v4,v3已经被标记不能相互连接,避免产生回路,从v1、v4、v3中可选择的边有:
(v1,v2)权值为6
(v4,v5)权值为10
(v3,v2)权值为2
从中选择权值最小为2的边,(v3,v2)
在这里插入图片描述

从v1,v4,v3,v2中可选择的边有:
(v2,v9)权值为1
(v4,v5)权值为10
从中选择权值最小1的边,(v2,v9)
在这里插入图片描述

从v1,v4,v3,v2,v9中可选边有:
(v4,v5)权值为10
(v9,v5)权值为 6
(v9,v6)权值为 4
(v9,v7)权值为 3
(v9,v8)权值为 2
从中选择权值最小2的边,(v9,v8)
在这里插入图片描述

从v1,v4,v3,v2,v9,v8中可选边有:
(v4,v5)权值为10
(v9,v5)权值为 6
(v9,v6)权值为 4
(v9,v7)权值为 3
(v8,v7)权值为 3
从中有两条权值为3的边,任选一条(v8,v7)
在这里插入图片描述

从v1,v4,v3,v2,v9,v8,v7中可选边有:
(v4,v5)权值为10
(v9,v5)权值为 6
(v9,v6)权值为 4
(v7,v6)权值为 4
从两条权值为4的边,任选一条(v7,v6)
在这里插入图片描述

从v1,v4,v3,v2,v9,v8,v7,v6中可选边有:
(v4,v5)权值为10
(v9,v5)权值为 6
(v6,v5)权值为 2
选择最小权值为2的边(v6,v5)
在这里插入图片描述

v1,v4,v3,v2,v9,v8,v7,v6,v5所有点被标记
最小生成树找到,总权值为17
在这里插入图片描述

C语言Prim算法实现

#include<stdio.h>
#define M 1000//M表示无穷用1000代替
//判断标记点的函数,避免产生回路
int Is(int arr[9], int flag)
{
	int i = 0;
	for (i = 0; i < 9; i++)
	{
		if (arr[i] == flag)
			return 0;
	}
	return 1;
}
int main()
{
	//把上图权值对应值写成邻接阵
	int map[9][9] =
	{
		{M,6,3,1,M,M,M,M,M},
		{6,M,2,M,M,M,M,M,1},
		{3,2,M,2,M,M,M,M,M},
		{1,M,2,M,10,M,M,M,M},
		{M,M,M,10,M,2,M,M,6},
		{M,M,M,M,2,M,4,M,4},
		{M,M,M,M,M,4,M,3,3},
		{M,M,M,M,M,M,3,M,2},
		{M,1,M,M,6,4,3,2,M}
	};

	//存放被标记的点
	int arr[9] = { 1 };//设置初始点为V1,只有V1被标记
	//记录标记个数
	int count = 1;
	//存放权值总和
	int s = 0;
	//循环变量
	int i = 0;
	//记录最小权下标
	int index = 0;
	//记录最小权
	int min = M;
	//记录点
	int doc = 1;

	//循环部分:
	while (1)
	{

		min = M;
		for (i = 0; i < count; i++)
		{
			int j = 0;
			int c = arr[i] - 1;
			for (j = 0; j < 9; j++)
			{
				if (map[c][j] <= min && Is(arr, j + 1))
				{
					doc = c + 1;
					min = map[c][j];
					index = j;
				}
			}
		}
		s += min;
		arr[count++] = index + 1;
		//打印
		printf("V%d --> V%d ", doc, index + 1);
		//所有点被标记,跳出循环
		if (count == 9)
			break;

	}
	printf("\n总权值为:%d\n", s);
	return 0;
}

运行结果:

在这里插入图片描述

2、用Prim-Alternate算法求最小生成树

Prim-Alternate算法是Prim算法的优化
Prim-Alternate算法是只找当前标记点和上一个标记点的邻边
被标记的点不能互相相连

3、C语言Prim-Alternate算法实现

//Prim-Alternate算法
#include<stdio.h>
#define M 1000//M表示无穷用1000代替
//判断标记点的函数,避免产生回路
int Is(int arr[9], int flag)
{
	int i = 0;
	for (i = 0; i < 9; i++)
	{
		if (arr[i] == flag)
			return 0;
	}
	return 1;
}
int main()
{
	//把上图权值对应值写成邻接阵
	int map[9][9] =
	{
		{M,6,3,1,M,M,M,M,M},
		{6,M,2,M,M,M,M,M,1},
		{3,2,M,2,M,M,M,M,M},
		{1,M,2,M,10,M,M,M,M},
		{M,M,M,10,M,2,M,M,6},
		{M,M,M,M,2,M,4,M,4},
		{M,M,M,M,M,4,M,3,3},
		{M,M,M,M,M,M,3,M,2},
		{M,1,M,M,6,4,3,2,M}
	};

	//存放被标记的点
	int arr[9] = { 1 };//设置初始点为V1,V1被标记
	//记录标记个数
	int count = 1;
	//存放权值总和
	int s = 0;
	//循环变量
	int i = 0;
	//记录最小权下标
	int index = 0;
	//记录最小权
	int min = M;
	//记录点
	int doc = 1;

	int c = 0;
	//循环部分:
	while(1)
	{
		
		 min = M;
//每次循环只找两个标记点相邻的边
		for(i=count-2;i< count ;i++)
		{ 
			int j = 0;
			if (count == 1)
				c = arr[0] - 1;
			else
				c = arr[i] - 1;
			for (j = 0; j < 9; j++)
			{
				if (map[c][j] <= min && Is(arr, j + 1))
				{
					doc = c+1;
					min = map[c][j];
					index = j;
				}
			}
		}
		s += min;
		arr[count++] = index + 1;
		//打印
		printf("V%d --> V%d ",doc , index + 1);
		//所有点被标记,跳出循环
		if (count == 9)
			break;		
	}
	printf("\n总权值为:%d\n", s);
	return 0;
}

运行结果:

在这里插入图片描述

  • 19
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值