首页 > 编程语言 >C++ 图论之次最小生成树

C++ 图论之次最小生成树

时间:2023-11-29 11:36:59浏览次数:48  
标签:pos 之次 权重 int 图论 C++ 生成 最小 节点

1. 前言

生成树指在无向图中找一棵包含图中的所有节点的树,此树是含有图中所有顶点的无环连通子图。对所有生成树边上的权重求和,权重和最小的树为最小生成树,次小的为次最小生成树。

最小生成树和次最小生成树的应用领域都较广泛。也是图论中优为重要的研究对象,求解算法也是常规必须掌握的算法之一。最小生成树的算法主要是kruskal(克鲁斯卡尔)Prim算法,求解次最小生成树时也是基于这两种算法,在此之上略做些变化。

2. 次最小生成树算法

2.1 完全穷举法

基本思想,先找出无向图中的最小生成树,依次删除最小生成树上的一条边,再在图中找最小生成树,会得到值不同的最小生成树,取权重和最小的即次最小生成树。

流程如下:

  • 首先在图中找到最小生成树,如下图红色标记的边所组成的树,其权重和为 30

1.png

  • 从最小生成树中删除一条边,然后再在图中查找最小生成树。

2.png

  • 重复上述流程,从而找到次最小生成树。如下图为删除权重为3的边和权重为4的边后的最小生树的权重和。

3.png

4.png

  • 如下为次最小生树。

5.png

时间复杂度分析。

prim最小生成树算法的时间算法度(n表示节点数,e表示边数):

  • 邻接矩阵O(n<sup>2</sup>)
  • 邻接表 O(elog<sub>2</sub>n)

则基于prim的次最小生成树的时间复杂度为:

  • 邻接矩阵O(n<sup>3</sup>)
  • 邻接表 O(nelog<sub>2</sub>n)

完全暴力穷举法的时间复杂度较高,算是一种思路,但绝对不适用。

2.2 先添后删

基本思想,使用prim或者kruskal算法找出最小生成树。则图上的边分属于两个集合,最小生成树集合(下称 t)和非最小生成树集合(下称t1)。如下图,红色的边是最小生成树上的边,蓝色边为非最小生成树上的边,或者说是被排除的边。

1.png

最小生成树和次最小生成树的权重的差异一定是一对边的差异,这一对边中的一条在t上,一条在t1上。可以把最小生成树和上文使用穷举法找出来的次最小生成权做对比。

也就是一旦把最小生成树(5,7,5)这条边换成(6,7,6)这条边,便找到了次最小生成树。当然,当前是在已知情况推导结论,过程是需要算法来支撑。

Tips: (5,7,5)其中的第一个和第二个数字表示节点编号 ,第三个数字表示连接第一个和第二个节点的边的权重。

6.png

先添后删算法的基本原则:就是用t1中的边替换t中的边。

显然,替换要讲究策略。比如说(3,4,12)t1上的边,具体应该替换t上的那一条边?如果逐一替换,除了时间复杂度也会较高,也会面临一个问题,不是每一次替换后会重新得到一棵树。

先添后删是有策略的。

  • 如下图,先从t1集合中找出(3,4,12)的边,并添加到最小生成树上。这时发现1,2,3,4四个节点构建成了一个环(回路)。原理很简单,在树上的非边上的任意两点间连一条线,都将会出现经过这两点的环。

7.png

  • 添加完毕,需从环上删除一条边,只有这样,方能重新变成树。删除策略,把环上除了新添加边之外权重最大的边删除。如下图,删除环上权重为5的边。

    删除后新树的权重和=原最小生成树权重和+新添加的边的权重-删除的边的权重。

    如下图,权重和为=30+12-5=37。原最小生树的权重和是固定的,所以我们要找的是新添加的边的权重-删除边的权重值(增量)最小的那次替换。

8.png

  • 有了方向后,完成所有的替换过程,并且查找此过程中增量值最小的那次替换。如下图是添加 (4,5,9),删除(4,8,8),增量=9-8=1

9.png

  • 如下图是添加(7,8,20),删除(5,7,5),增量=20-5=15

10.png

  • 下图是添加 (6,7,6),删除(5,7,5),增量=6-5=1

11.png

通过推导得知,次最小生成树的权重和为31

如果是求严格的次小生成树的,在环路去边的步骤中,如果回路中除当前边之外权值最大的边的权值等于当前边的权值,那就去掉回路中除当前边之外权值第二大的边。

2.3 编码中的难点

求解次最小生成树,需要先求解最小生成树,这个简单,使用prim或者kruskal算法便可。摆在面前的主要问题是,在添加一条边后所构建成的环上,如果找到权重最大的边,并删除它。

这里使用动态规划思想,记录最小生成树上任意两点间权重最大的边。

  • 构建名为dp 的矩阵。行号i和列号j表示最小生成树上任意两点的编号。值为两点间权重最大的边的值。

12.png

  • 推导动态转移方程式。按prim算法思路,绘制出最小生成树上每一边出现的顺序。如下图所示,当选择了3号节点后,dp[1][3]dp[1][2]的值可以很容易推导出来。dp[2][3]的值直观上也可以得到,在 dp[1][3]dp[1][2]中选择值较大即可。

    Tips:1号节点的父节点编号设为0

14.png

  • 那么通用的动态转移方程式应该如何书写?

    把求解dp[2][3]的值分成两段来求。

    节点1是节点3的父节点,节点3被选择出来后,它与父节点的权重是可知,即为5,再求父节点1和节点2之间的最大权重边的值(树是连通的,节点 3 一定是可以通过父节点到达 2节点)。再在两者中取最大值。说起来有点绕口,看一下图解。

15.png

  • 可以得到通用的转移方程式,dp[i][j]=max(dp[i][i父节点],dp[i的父节点][j])

具体代码:

#include <bits/stdc++.h>
using namespace std;
const int INF=999;
//矩阵
int  graph[100][100];
//动态规划表
int maxWeight[100][100];
//  i - j 边是否是在树上还是在非树上
bool used[100][100];
//前驱
int pre[100];
//距离
int dis[100];
//节点是否访问,构建最小生成树的过程中使用
int vis[100];
//点数,边数
int n,m,minWeight=0;

fstream cin_;

void init() {
	for(int i=1; i<=n; i++) {
		for(int j=1; j<=n; j++) {
			maxWeight[i][j]=0;
			used[i][j]=0;
			if(i==j)graph[i][j]=0;
			else graph[i][j]=INF;
		}
	}
}
void read() {
	int f,t,w;
	for(int i=0; i<m; i++) {
		cin>>f>>t>>w;
		graph[f][t]=graph[t][f]=w;
	}
}
//最小生成树
void prim(int start) { //start=1
	//初始化 dis 存储其它节点到 start(1) 节点的权重(距离)
	for(int i=1; i<=n; i++) {
		dis[i]=graph[start][i];
		//所有节点的父节点都是 start
		pre[i]=start;
	}
	// 确定第一个节点
	vis[start]=true;
	pre[start]=0;
	//找 n-1 次
	for(int j=0; j<n-1; j++) {
		int minVal,pos;
		minVal=INF;
		pos=-1;
		//找最小值 dis
		for(int i=1; i<=n; i++) {
			if( vis[i]==0 && dis[i]<minVal ) {
				minVal=dis[i];
				pos=i;
			}
		}
		//
		vis[pos]=true;
		minWeight+=dis[pos]; //minVal ===dis[pos]
		//边置为访问
		used[pos][ pre[pos] ]=  used[ pre[pos]  ][ pos ]=1;
		//pos 激活点  -->点之间
		for(int i=1; i<=n; i++) {
			if(vis[i]==1 ) {
				//分成两段,先求自己和父节点的权重,再求父节点到指定节点的最大权重
				maxWeight[pos][i]=	maxWeight[i][pos]=max( dis[pos],maxWeight[i][ pre[pos]  ]  );
			}

			if( vis[i]==0 && graph[pos][i]<dis[i] ) {
				dis[i]=graph[pos][i];
				pre[i]=pos;
			}
		}
	}
}
//次最小生成树
int ctree() {
	int val=INF;
	for(int i=1; i<=n; i++ ) {
		for( int j=i+1; j<=n; j++ ) {
			if( used[i][j]==0 &&  graph[i][j]!=INF ) {
				val=min (val,  minWeight+ graph[i][j]-maxWeight[i][j] );
			}
		}
	}
	return val;
}

int main() {
//	freopen("min.in","r",stdin);
	cin.open("min.in",ios::in);
	cin>>n>>m;
	init();
	read();
	prim(1);
	cout<<minWeight<<endl;
	int res= ctree();
	cout<<res<<endl;
	return 0;
}

3. 严格次最小生成树

如果添加的边的权重和环上最大边的权重相同,这时删除最大边的权重和没有删除是没有区别的,或者说,这时得到的次最小生成树并不是严格前意义上的次最小生成树,得到的次最小生成树有可能和最小生成树是一样。

如何解决这个问题?

记录树上任意两点间的路径上的最大边权重值时,同时也记录第二大权重值。则dp二维数组需改成三维数组。

dp[i][j][0]缓存树上任意两点间的所有边中权重最大的值,dp[i][j][1]缓存所有边中权重第二大的值。

如下仅显示需要修改的代码。

void prim(int start) { 
//……省略    
maxWeight[pos][i]=	maxWeight[i][pos]=max( dis[pos],maxWeight[i][ pre[pos]  ]  );
				if( dis[pos]>maxWeight[i][ pre[pos][0] ){
					//第一大变成第二大
					maxWeight[pos][i][1]=maxWeight[pos][i][0];
					maxWeight[pos][i][0]=dis[pos];
				}else if(  dis[pos]>maxWeight[i][ pre[pos][1] )
				     maxWeight[i][ pre[pos][1]= dis[pos];
//……省略
}   
int ctree() {
	int val=INF;
	for(int i=1; i<=n; i++ ) {
		for( int j=i+1; j<=n; j++ ) {
			if( used[i][j]==0 &&  graph[i][j]!=INF ) {
                 if(graph[i][j]-maxWeight[i][j][0]==0){
                     val=min (val,  minWeight+ graph[i][j]-maxWeight[i][j][1] );
                 }else{
                      val=min (val,  minWeight+ graph[i][j]-maxWeight[i][j][0] );
                 }
			}
		}
	}
	return val;
}

4.总结

在一棵树中,连接两个不在树上任意边两端的点,则会在这两个点之间形成一个环,然后通过最小生成树和次最小生成树的大小差异一定是一对边差异特性,很方便求解出最终答案。

本文已收录到"编程驿站"公众号。

标签:pos,之次,权重,int,图论,C++,生成,最小,节点
From: https://blog.51cto.com/gkcode/8612819

相关文章

  • C/C++ 常用的四种查找算法
    在计算机科学中,搜索算法是一种用于在数据集合中查找特定元素的算法。C语言作为一种强大的编程语言,提供了多种搜索算法的实现方式。本文将介绍C语言中的四种常见搜索算法其中包括(线性查找,二分法查找,树结构查找,分块查找),并提供每种算法的简单实现示例。常见的查找算法主要有以下几种......
  • C++容器中存放的是数据本身还是数据地址?
    在C++中,std::map容器内存放的是数据本身(即键值对的值部分),而不是数据地址。当我们插入一个键值对时,std::map会自动复制值并存储副本voidtest02(){ multimap<int,Worker>m; Workerw; w.name="sd"; w.salary=1234; m.insert(pair<int,Worker>(1,w)); multimap<i......
  • c++ deque容器
    一、deque介绍deque(双端队列)是一种索引容器,它包含在#include<deque>头文件中。它与普通的queue队列不同的是,deque可以实现在尾部插入和删除元素。随机的访问双端队列中的元素,时间复杂度为O(1)在首部或者尾部插入或删除元素,时间复杂度O(1)插入和删除元素,是线性的,时间复杂度为O......
  • C++中的红黑树学习
    C++中有一种数据结构-红黑树,在C++的STL中有一种数据结构map,它就是基于红黑树来实现的红黑树,是一种二叉搜索树,但是它的每个节点都有颜色,并且只有红和黑两种颜色。所以每个节点上都有一个存储位来表示节点的颜色,可以是Red和Black.红黑树有一个很大的特点: 它能够确保没有任何一......
  • C++ Primer 学习笔记——第十三章
    第十三章拷贝控制前言类是如何控制类型对象的拷贝、赋值、移动和销毁的?类通过一些特殊的成员函数控制,包括:拷贝构造函数、移动构造函数、拷贝赋值运算符、移动赋值运算符以及析构函数。当定义一个类时,我们显式地或隐式的指定在此类型的对象拷贝、移动、赋值和销毁时做什么。一......
  • C++17 更通用的 union:variant
    References现代C++学习——实现多类型存储std::variant如何优雅的使用std::variant与std::optionalstd::variant是C++17中,一個新加入標準函式庫的template容器;他的概念基本上是和union(參考)一樣,是一個可以用來儲存多種型別資料的容器。比如說:std::variant<int,d......
  • C++U4-第06课-二分答案
    上节课作业解析链接:https://pan.baidu.com/s/1QCDg1GXb5HhrpkPgomOCyg?pwd=s4b4提取码:s4b4二分答案学习目标二分查找单调性意思 二分答案单调性 二分答案的思路[【二分答案】砍树(简单版)]枚举每一棵树,注意当锯片高度高于树的高度时砍的树木是0。#include<io......
  • C++ bool 类型
    @TOC一.bool类型在C++中,bool类型用于表示逻辑值,它只有两个可能的取值:true(真)和false(假)。bool类型常用于条件判断和布尔运算中。C++标准要求bool类型占用一个字节的内存空间。它的取值只能是true或false,并且可以通过关键词true和false直接赋值。下面是一些常见的使......
  • C++获取机器启动至今的时长和机器启动的时间戳
    根据当前时间戳与机器启动至今的时间长度相减,可以精确计算出机器启动时刻的时间戳epochtime代码#include<iostream>#include<stdio.h>#include<time.h>#include<chrono>intmain(){#ifdef__linux //linuxonly std::cout<<"===linuxonlytimeanalysis==......
  • C/C++ Zlib实现文件压缩与解压
    在软件开发和数据处理中,对数据进行高效的压缩和解压缩是一项重要的任务。这不仅有助于减小数据在网络传输和存储中的占用空间,还能提高系统的性能和响应速度。本文将介绍如何使用zlib库进行数据的压缩和解压缩,以及如何保存和读取压缩后的文件。zlib是一个开源的数据压缩库,旨在提......