热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

数据结构-图详解(图基本概念、图的存储结构及C++实现)

本文主要介绍关于数据结构,c++,图论的知识点,对【数据结构-图详解(图基本概念、图的存储结构及C++实现)】和【数据结构图的存储结构代码】有兴趣的朋友可以看下由【NUC_Dodamce】投稿的技术文

本文主要介绍关于数据结构,c++,图论的知识点,对【数据结构-图详解(图基本概念、图的存储结构及C++实现)】和【数据结构图的存储结构代码】有兴趣的朋友可以看下由【NUC_Dodamce】投稿的技术文章,希望该技术和经验能帮到你解决你所遇的# 图论相关技术问题。

数据结构图的存储结构代码

文章目录 1.图的基本概念有向图和无向图完全图邻接顶点顶点的度路径子图、连通图、强连通图生成树 2.图的存储结构图的创建邻接矩阵C++邻接矩阵创建图 邻接表C++邻接表创建图

1.图的基本概念

图是由顶点集合及顶点间的关系组成的一种数据结构.
表示为 G = (V, E)。V代表的是顶点集合,E代表边集合

树也是一种特殊的图,这个图无环.
树关注的时节点存的值,图更关注的是顶点及边的权值。
权值:边中的附带的数据信息.

顶点和边:图中结点称为顶点,第i个顶点记作vi。两个顶点vi和vj相关联称作顶点vi和顶点vj之间有一条边,图中的第k条边记作ek,ek = (vi,vj)或 .

有向图和无向图

有向图:顶点对 是有序的,顶点对 称为顶点x到顶点y的一条边(弧), 是两条不同的边。
eg:

数据结构-图详解(图基本概念、图的存储结构及C++实现)


无向图:顶点对(x, y)是无序的,顶点对(x,y)称为顶点x和顶点y相关联的一条边,这条边没有特定方向,(x, y)和(y,x)是同一条边。

数据结构-图详解(图基本概念、图的存储结构及C++实现)

完全图

在有n个顶点的无向图中,若有n * (n-1)/2条边,即任意两个顶点之间有且仅有一条边,则称此图为无向完全图;在n个顶点的有向图中,若有n * (n-1)条边,即任意两个顶点之间有两条边,且仅这两条边方向相反,则称此图为有向完全图。

邻接顶点

无向图中:两个顶点U、V有一条边连接。成为U、V互为临界顶点。

有向图中:两个顶点U、V。若是U指向V,则称为U邻接到V,V邻接自顶点U。并称这条边与顶点U、V相关联。

顶点的度

顶点V的度指的是与它相关联的边的条数。

特殊:

对于有向图,顶点的度等于该顶点的入度+出度。
出度:从顶点出去的边的条数。
入度:指向这个顶点的边数。对于无向图:入度=出度=顶点的度 路径

从顶点U到顶点V的一组边称为路径。路径不止一条。

路径长度:对于不带权图为路径的边个数。带权图为路径所有边权值的和

最短路径:所有路径,路径长度最小的路径。

简单路径与回路:

简单路径:路径上经过顶点不重复。回路:路径上经过的顶点有重复。 子图、连通图、强连通图

两张图A、B,若A的顶点时B的部分顶点,A的边是B的部分边,则称为A是B的子图。

连通图:无向图中,若顶点A、B存在路径,称为A、B连通。若图中的任意两点都是连通的,则称此图为连通图。

强连通图:有向图中,若图中的任意两点都是连通的(A指向B,且B指向A),则称此图为连通图。

生成树

无向图中一个连通图的最小连通子图称为生成树。(用最少的边把所有顶点连接起来)。n个顶点的连通图的生成树有n-1条边。

最小生成树:所有生成树中,路径长度最小的生成树。

2.图的存储结构

因为图中既有节点,又有边(节点与节点之间的关系),因此,在图的存储中,只需要保存:节点和边的关系即可。

节点通过数组保存,边的关系通过下面的方式保存

图的创建

可以通过输入的方式来创建,或者是通过文件的方式来读取。

这里采用手动添加边来创建图,方便测试.

邻接矩阵

邻接矩阵(二维数组)即是:先用一个数组将定点保存,然后采用矩阵来表示节点与节点之间的关系。

eg:

数据结构-图详解(图基本概念、图的存储结构及C++实现)

如果矩阵位置为0代表无关,位置为1代表连接。A对应的下标为0,D对应的下标为3。这里采用map映射的方式来建立顶点与顶点下标的关系.
容易得出:无向图为对称矩阵。

此外:如果这个图是带权图,则矩阵内的数据为所带权值

临界矩阵的存储方式的优点:

非常适合边比较多的情况。O(1)判断两个节点的连接关系。

不足:

O(N)查找一个节点连接的所有边。(N为节点个数)。 C++邻接矩阵创建图
#pragma once

#include
    
#include
#include
    

//v:顶点 w:权值 max_w:最大权值 Direction:判断图是有向图还是无向图

namespace matrix {
   
	//邻接矩阵保存边关系
	template<class v, class w, w max_w = INT_MAX, bool Direction = false>
	class Graph {
   
	private:
		std::vector<v>_vertexs;//顶点集合
		std::map<v, int>_indexMap;//顶点与下标的映射
		std::vector<std::vector<w>>_matrix;//邻接矩阵
				//获取顶点下标
		size_t GetPointIndex(const v& point) {
   
			auto ptr = _indexMap.find(point);
			if (ptr != _indexMap.end()) {
   
				return ptr->second;
			}
			else {
   
				throw std::invalid_argument("顶点不存在");
				return -1;
			}
		}
	public:
		//图的创建
		Graph(std::vector<v>& points) {
   
			_vertexs.resize(points.size());
			for (size_t i = 0; i < points.size(); i++) {
   
				_vertexs[i] = points[i];
				_indexMap[points[i]] = i;
			}

			_matrix.resize(points.size());
			//邻接矩阵
			for (int i = 0; i < _matrix.size(); i++) {
   
				_matrix[i].resize(points.size(), max_w);
			}
		}

		//添加边关系,输入两个点,以及这两个点连线边的权值。
		void AddEdge(const v& src, const v& det, const w& weight) {
   
			size_t posSrc = GetPointIndex(src);
			size_t posDet = GetPointIndex(det);
			//区分有向图与无向图
			_matrix[posSrc][posDet] = weight;
			if (Direction == false) {
   
				//无向图,添加两条边关系
				_matrix[posDet][posSrc] = weight;
			}
		}

		void Print() {
   
			//打印顶点对应坐标
			for (size_t i = 0; i < _vertexs.size(); i++) {
   
				std::cout << "[" << i << "]" << "->" << _vertexs[i] << std::endl;
				std::cout << std::endl;
			}

			//打印边
			std::cout << " ";
			for (int i = 0; i < _vertexs.size(); i++) {
   
				std::cout << _vertexs[i] << " ";
			}
			std::cout << std::endl;

			//打印矩阵
			for (size_t i = 0; i < _matrix.size(); i++) {
   
				std::cout << _vertexs[i] << " ";
				for (size_t j = 0; j < _matrix[i].size(); j++) {
   
					if (_matrix[i][j] == max_w) {
   
						std::cout << "*" << " ";
					}
					else {
   
						std::cout << _matrix[i][j] << " ";
					}
				}
				std::cout << std::endl;
			}
		}
	};
}
#include"Graph.h"

using namespace std;

void TestGraph() {
   
	vector<char>points = {
    'A','B','C','D' };
	matrix::Graph<char, int, INT_MAX, true>graph(points);
	graph.AddEdge('A', 'B', 1);
	graph.AddEdge('A', 'D', 4);
	graph.AddEdge('B', 'D', 2);
	graph.AddEdge('B', 'C', 9);
	graph.AddEdge('C', 'D', 8);
	graph.AddEdge('C', 'B', 5);
	graph.AddEdge('C', 'A', 3);
	graph.AddEdge('D', 'C', 6);
	graph.Print();
}

int main() {
   
	TestGraph();
}

数据结构-图详解(图基本概念、图的存储结构及C++实现)

邻接表

使用数组表示顶点的集合,使用链表表示边的关系。

创建指针数组,自己连接的顶点挂在下面。同样的,对节点进行编号

数据结构-图详解(图基本概念、图的存储结构及C++实现)


邻接表的优点:

适合保存稀疏的边关系。适合查找一个顶点连接出的边

不足:

不适合确定两个顶点是否相连,判断权值。

注意:如果图是有向图,则如果使用邻接表存储边的关系,需要保存出边表和入边表。

C++邻接表创建图
#pragma once

#include
    
#include
#include
    

//v:顶点 w:权值 max_w:最大权值 Direction:判断图是有向图还是无向图

//邻接表保存边关系
namespace link_table {
   
	template<class w>
	struct Edge{
   
		int detPos;
		int srcPos;
		w weight;
		Edge<w>* next;
		Edge(int _srcPos, int _detPos, const w& _weight)
			:detPos(_detPos), srcPos(_srcPos), weight(_weight),next(nullptr)
		{
   }
	};
	template<class v, class w, bool Direction = false>
	class Graph {
   
		typedef Edge<w> Edge;
	private:
		std::vector<v>_vertexs;//顶点集合
		std::map<v, int>_indexMap;//顶点与下标的映射
		std::vector<Edge*>_tables;//邻接表
		//获取顶点下标
		size_t GetPointIndex(const v& point) {
   
			auto ptr = _indexMap.find(point);
			if (ptr != _indexMap.end()) {
   
				return ptr->second;
			}
			else {
   
				throw std::invalid_argument("顶点不存在");
				return -1;
			}
		}
	public:
		//图的创建
		Graph(std::vector<v>& points) {
   
			_vertexs.resize(points.size());
			for (size_t i = 0; i < points.size(); i++) {
   
				_vertexs[i] = points[i];
				_indexMap[points[i]] = i;
			}
			_tables.resize(points.size(), nullptr);
		}

		//添加边关系,输入两个点,以及这两个点连线边的权值。
		void AddEdge(const v& src, const v& det, const w& weight) {
   
			size_t posSrc = GetPointIndex(src);
			size_t posDet = GetPointIndex(det);

			Edge* edge = new Edge(posSrc, posDet, weight);
			//头插
			edge->next = _tables[posSrc];
			_tables[posSrc] = edge;

			if (Direction == false) {
   
				//有向图
				Edge* edge = new Edge(posDet, posSrc, weight);
				edge->next = _tables[posDet];
				_tables[posDet] = edge;
			}
		}

		void Print() {
   
			//打印顶点对应坐标
			for (size_t i = 0; i < _vertexs.size(); i++) {
   
				std::cout << "[" << i << "]" << "->" << _vertexs[i] << std::endl;
				std::cout << std::endl;
			}

			for (size_t i = 0; i < _tables.size(); i++) {
   
				std::cout << _vertexs[i] << "[" << i << "]:";
				Edge* node = _tables[i];
				while (node != nullptr) {
   
					std::cout << _vertexs[node->detPos] <<
						"[" << node->detPos << "]" << "(" << node->weight << ")" << " ";
					node = node->next;
				}
				std::cout << "nullptr" << std::endl;
			}
		}
	};
}
void TestGraph2() {
   
	vector<char>points = {
    'A','B','C','D' };
	link_table::Graph<char, int, true>graph(points);
	graph.AddEdge('A', 'B', 1);
	graph.AddEdge('A', 'D', 4);
	graph.AddEdge('B', 'D', 2);
	graph.AddEdge('B', 'C', 9);
	graph.AddEdge('C', 'D', 8);
	graph.AddEdge('C', 'B', 5);
	graph.AddEdge('C', 'A', 3);
	graph.AddEdge('D', 'C', 6);
	graph.Print();
}

int main() {
   
	//TestGraph();
	TestGraph2();
}

数据结构-图详解(图基本概念、图的存储结构及C++实现)

本文《数据结构-图详解(图基本概念、图的存储结构及C++实现)》版权归NUC_Dodamce所有,引用数据结构-图详解(图基本概念、图的存储结构及C++实现)需遵循CC 4.0 BY-SA版权协议。


推荐阅读
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
  • 本文详细介绍了C语言中链表的两种动态创建方法——头插法和尾插法,包括具体的实现代码和运行示例。通过这些内容,读者可以更好地理解和掌握链表的基本操作。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 文件描述符、文件句柄与打开文件之间的关联解析
    本文详细探讨了文件描述符、文件句柄和打开文件之间的关系,通过具体示例解释了它们在操作系统中的作用及其相互影响。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • Splay Tree 区间操作优化
    本文详细介绍了使用Splay Tree进行区间操作的实现方法,包括插入、删除、修改、翻转和求和等操作。通过这些操作,可以高效地处理动态序列问题,并且代码实现具有一定的挑战性,有助于编程能力的提升。 ... [详细]
author-avatar
真理难辩_175
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有