图结构系列—基于邻接矩阵的图实现

在数据结构中跌跌撞撞的走到了图这个部分,回想在几个星期前还在为各种树的各种结构定义与算法实现苦恼,忽然间已经来到了数据结构中貌似是最重要的一个结构了。在之前就有听说图在各行各业的理论研究上用途很广,如人工智能,寻路,更让我想不到的还有化合物的鉴定......可见图的未知的应用还有多少了。

从逻辑上分,图分为有向图与无向图。从存储表示上可以有着重用顶点表示一个图的,当然也有着重用边来表示一个图的。我们现在只学习了着重用顶点来表示一个有向图,下面就来详述。

用顶点表示的图又可以分为基于邻接矩阵的形式,基于邻接表的形式。他们之间各有利弊吧,比如在游戏编程中用到的用于处理各种逻辑先后顺序,一般用的是基于邻接矩阵的图,这种一般用于图不经常动的情况下。如果一个图经常的要对定点,边信息进行更新的话,那么还是用基于邻接表的图更好一些,因为链表本来易于更新。

在这里实现的是一个基于邻接矩阵标志的带权有向图,用到了模板,顶点是一种类型,边的信息又是一种类型,所以出现了template<class T,class E>的形式。用基于邻接矩阵的表示可以很方便的表示一个有向图,只需把矩阵做成不对称的就行了。但是如果是基于邻接表的存储方式,那么要出现一个出表,一个如表,在表示有向图上麻烦了许多。这个在后序文章中详细叙述。

先来看代码:

#pragma once
#include<iostream>
#include<fstream>
using namespace std;
const int DefaultVerticesNum = 30;

template<class T, class E>
class GraphAdjMatrix
{
public:
        GraphAdjMatrix(const E &maxweight, int size = DefaultVerticesNum); //构造一个空图 配置用于表示两点之间不可达的信息
	~GraphAdjMatrix(); //销毁图

	T GetVertex(int vertex_idx); //根据顶点索引返回顶点元素值 也会是自定义的类对象
	E GetWeigth(const T &begvtx, const T &endvtx); //获取目标对象begvtx,endvtx之间的权重值
	T GetFirstNeighbor(const T &vertex); //获取以vertex为顶点的第一个邻接顶点
	T GetNextNeighbor(const T &vertex, const T &adj_vertex); //获vertex的邻接顶点adj_vertex的下一个邻接顶点
	
	bool InsertVertex(const T &vertex); //插入一个顶点
	bool InsertEdge(const T &begvtx, const T &endvtx, const E &cost); //在顶点begvtex,endvtx之间插入一条边(插入权重值)
	bool RemoveVertex(const T &vertex); //删除一个顶点
	bool RemoveEdge(const T &begvtx, const T &endvtx); //删除begvtx,endvtx两顶点之间的边

	template<class T, class E> friend istream& operator>>(istream &input, GraphAdjMatrix<T,E> &G); //重载输入
	template<class T, class E> friend ostream& operator<<(ostream &output, GraphAdjMatrix<T,E> &G); //重载输出
	template<class T, class E> friend ifstream& oper
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
邻接矩阵是一种表示的方式,它用二维数组来表示节点之间的连接关系。具体来说,如果中有 $n$ 个节点,那么邻接矩阵就是一个 $n \times n$ 的矩阵,其中第 $i$ 行第 $j$ 列的元素表示从节点 $i$ 到节点 $j$ 是否有边相连。 广度优先遍历是一种的遍历方式,它从中的某个节点开始,依次访问与该节点相邻的所有节点,然后再依次访问这些相邻节点的相邻节点,直到遍历完整个。 下面是基于邻接矩阵实现的广度优先遍历的代码: ```python from queue import Queue def bfs(adj_matrix, start): n = len(adj_matrix) visited = [False] * n queue = Queue() visited[start] = True queue.put(start) while not queue.empty(): node = queue.get() print(node, end=' ') for i in range(n): if adj_matrix[node][i] and not visited[i]: visited[i] = True queue.put(i) ``` 其中,`adj_matrix` 是邻接矩阵,`start` 是起始节点的编号。首先创建一个长度为 $n$ 的布尔数组 `visited` 表示每个节点是否被访问过,初始时所有节点都未被访问过。然后创建一个队列 `queue`,将起始节点放入队列中,并将 `visited[start]` 设为 `True`。接下来进入循环,每次从队列中取出一个节点 `node`,访问它,并将与它相邻的未被访问过的节点放入队列中。具体来说,对于每个节点 $i$,如果从节点 $node$ 到节点 $i$ 有边相连且节点 $i$ 未被访问过,则将 `visited[i]` 设为 `True`,并将节点 $i$ 放入队列中。直到队列为空,遍历结束。 注意,上述代码中使用了 Python 中的队列类 `queue.Queue`,如果使用其他编程语言,需要自己实现队列。另外,如果不是连通的,即存在多个连通分量,那么需要对每个连通分量都进行广度优先遍历。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值