数据结构复习笔记6.2:图的存储和遍历

news2025/1/11 1:48:42

图在内存中存储⽅式有很多种,最经典的包括邻接矩阵、邻接表、逆邻接表和⼗字链表。

1.图的存储

1.1邻接矩阵

图的邻接矩阵是⽤两个数组来表示,⼀个⼀维数组存储图中的顶点信息,⼀个⼆维数组(我们将这 个数组称之为邻接矩阵)存储图中的边的信息。

1.1.1⽆向图邻接矩阵

我们可以设置两个数组,顶点数组为vertex[4]={V0,V1,V2,V3},边数组arc[4] [4]为对称矩阵(0表示 不存在顶点间的边, 1表示顶点间存在边)。

对称矩阵

所谓对称矩阵就是n阶矩阵的元素满⾜a[i] [j] = a[j] [i] (0<=i,j<=n)。即从矩阵的左上⻆到右下⻆的主 对⻆线为轴,右上⻆的元与左下⻆相对应的元全都是相等的。

1.1.2有向图邻接矩阵

⽆向图的边构成了⼀个对称矩阵,貌似浪费了⼀半的空间,那如果是有向图来存放,会不会把资源 都利⽤得很好呢?

1.1.3带权图(网)的邻接矩阵

带权图中的每⼀条边上带有权值,邻接矩阵中的值则为权值,当两个顶点之间没有弧时,则⽤⽆穷 ⼤表示。

1.1.4优缺点

1.2邻接表

邻接表的处理⽅法是这样:

图中顶点⽤⼀个⼀维数组存储,当然,顶点也可以⽤单链表来存储,不过数组可以较容易地读取顶 点信息,更加⽅便。

图中每个顶点Vi的所有邻接点构成⼀个线性表,由于邻接点的个数不确定,所以我们选择⽤单链表 来存储。

1.2.1⽆向图邻接表

1.2.2有向图邻接表

1.2.3带权⽹络的邻接表

1.2.4优缺点

1.2.5代码展示
#include <iostream>
using namespace std;
/*
邻接表:一堆顶点结构体数组存点,用链表存边(结构体数组)
无向图两个点之间互相挂
有向图只挂出度的点,出度就统计该点链表长度,入度得遍历所有链表计算指向该点的个数
					如果只想找入度,就建立逆邻接表(入度挂在链表后面)看情况建立两种邻接链表

缺点:只能单独方便找出度(入度)
如果邻接表和逆邻接表结合->十字链表(有向图)
整合邻接表->多重邻接表(无向图)

*/
typedef struct EdgeNode
{
	char adj;//邻接点 可以是序号,可以就是本身数据,这里以数据为例
	struct EdgeNode *next;
	int w;//权值 ->图的权值是在边上
}Edge;
//顶点结构体数组
typedef struct VertexNode
{
	char data;
	struct EdgeNode * first;
}Vertex;
int n, m;//n是点的个数,m是无向边的个数
Vertex v[105];

int find(char x)
{
	for (int i = 1; i <= n; i++)
	{
		if (v[i].data == x)
		{
			return i;
		}
	}
}

int main()
{
	cin >> n >> m;
	//初始化每个点的数据和下标
	for (int i = 1; i <= n; i++)
	{
		cin >> v[i].data;
		v[i].first = NULL;
	}
	char x, y;
	int xi, yi;//x,y的下标
	int w;
	for (int i = 1; i <= m; i++)
	{
		cin >> x >> y >> w;//此示例为无向图,表示x和y有边
		Edge * s = new EdgeNode;
		s->adj = y;
		s->w = w;
		xi = find(x);
		s->next = v[xi].first;
		v[xi].first = s;

		Edge * s = new EdgeNode;
		s->adj = x;
		s->w = w;
		yi = find(y);
		s->next = v[yi].first;
		v[yi].first = s;

	}


	system("pause");
	return 0;
}
1.2.6邻接表与邻接矩阵的异同

1.3⼗字链表表示

⼗字链表的好处就是因为把邻接表和逆邻接表整合在了⼀起,这样既容易找到以Vi为尾的弧,也容 易找到以Vi为头的弧,因⽽容易求得顶点的出度和⼊度。

⼗字链表除了结构复杂⼀点外,其实创建图算法的时间复杂度是和邻接表相同的,因此,在有向图 的应⽤中,⼗字链表也是⾮常好的数据结构模型。

这个时候,还有⼀个问题,如果使⽤邻接表存储结构,但是对边的操作⽐较频繁,怎么办?

如果我们在⽆向图的应⽤中,关注的重点是顶点的话,那么邻接表是不错的选择,但如果我们更关注的是边的操作,⽐如对已经访问过的边做标记,或者删除某⼀条边等操作,邻接表的确显得不那么⽅便了。

代码展示:

#include <iostream>
using namespace std;
/*
如果邻接表和逆邻接表结合->十字链表(有向图)

整合邻接表->多重邻接表(无向图)
*/
//十字链表的结点结构
typedef struct ArcBox
{
	int tailvex;//弧尾
	int headvex;//弧头--->这里以下标为例
	struct ArcBox * tnext;//相同弧尾对应的链表的指针
	struct ArcBox * hnext;//相同弧头对应的链表的指针
	int w;

}Edge;
//图顶点的结构体数组
typedef struct
{
	int data;
	Edge * firstin;
	Edge * firstout;

}Vertex;

Vertex v[105];
int n, m;

void vertex_init()
{
	cin >> n >> m;
	for (int i = 1; i <= n; i++)
	{
		cin >> v[i].data;//数据是int类型的
		v[i].firstin = v[i].firstout = NULL;
	}
}

int main()
{
	vertex_init();
	int x, y, w;
	for (int i = 1; i <= m; i++)//输入边的关系(这边为了方便,不查找直接用的下标)
	{
		cin >> x >> y >> w;//  x(弧尾)--->y(弧头)
		Edge * s = new ArcBox;
		s->w = w;
		s->tailvex = x;
		s->headvex = y;

		s->tnext = v[x].firstout;
		v[x].firstout = s;//插入到弧尾的 出边链表中(该顶点的出度链表)

		s->hnext = v[y].firstin;
		v[y].firstin = s;//插入到弧头的 入边链表中(该顶点的入度链表)

	}

	system("pause");
	return 0;
}
1.4邻接多重表

邻接表对边的操作显然很不⽅便,因此,我们可以仿照⼗字链表的⽅式,对边表结构进⾏改装(将边的关系变为结点),重 新定义的边表结构如下:

其中iVex和jVex是与某条边依附的两个顶点在顶点表中的下标。 iLink指向依附顶点iVex的下⼀条 边, jLink指向依附顶点jVex的下⼀条边。

代码展示:

#include <iostream>
using namespace std;
/*
整合邻接表->多重邻接表(无向图)
*/
//多重邻接表的结点结构
typedef struct ArcBox
{
	int vi, vj;//结点下标
	struct ArcBox *inext;
	struct ArcBox *jnext;
	int w;

}Edge;
//图顶点的结构体数组
typedef struct
{
	int data;
	Edge * first;

}Vertex;

Vertex v[105];
int n, m;

void vertex_init()
{
	cin >> n >> m;
	for (int i = 1; i <= n; i++)
	{
		cin >> v[i].data;//数据是int类型的
		v[i].first= NULL;
	}
}

int main()
{
	vertex_init();
	int x, y, w;
	for (int i = 1; i <= m; i++)//输入边的关系(这边为了方便,不查找直接用的下标)
	{
		cin >> x >> y >> w;
		Edge * s = new ArcBox;
		s->w = w;
		s->vi = x;
		s->vj = y;

		//一个结点插入到两个链表里面 --》普通的邻接表就是开辟两个结点,每个first后面都连着对应的顶点下标(值本身)
		s->inext = v[x].first;
		v[x].first = s;
		s->jnext = v[y].first;
		v[y].first = s;
	
	}
	

	system("pause");
	return 0;
}
1.5边集数组

边集数组是由两个⼀维数组构成,⼀个是存储顶点的信息,另⼀个是存储边的信息,这个边数组每 个数据元素由⼀条边的起点下标(begin)、终点下标(end)和权(weight)组成。     

2.图的遍历   

遍历定义:从图中某顶点出发访遍图中所有顶点,且每个顶 点仅被访问一次,此过程称为图的遍历。

图的遍历算法是求解图的连通性问题、拓扑排序和求 关键路径等算法的基础

遍历实质:找每个顶点的邻接点的过程。

图的特点:图中可能存在回路,且图的任一顶点都可能与其它顶点相通,在访问完某个顶点之后可能会沿着某些边又回到了曾经访问过的顶点。

怎样避免重复访问?

解决思路:设置辅助数组visited[n],用来标记每个被访问过的顶 点。初始状态为0,i 被访问,改visited[i]为1,防止被多次访问

2.1深度优先遍历(DFS)
2.1.1方法总结

1、⾸先选定⼀个未被访问过的顶点V作为起始顶点(或者访问指定的起始顶点V),并将其 标记为已访问过;

2、然后搜索与顶点V邻接的所有顶点,判断这些顶点是否被访问过,如果有未被访问过的顶 点W;再选取与顶点W邻接的未被访问过的⼀个顶点并进⾏访问,依次重复进⾏。当⼀个顶点的所有的 邻接顶点都被访问过时,则依次回退到最近被访问的顶点。若该顶点还有其他邻接顶点未被访问,则从 这些未被访问的顶点中取出⼀个并重复上述过程,直到与起始顶点V相邻接的所有顶点都被访问过为⽌。

3、若此时图中依然有顶点未被访问,则再选取其中⼀个顶点作为起始顶点并进⾏遍历,转 ②。反之,则遍历结束。

2.1.2代码展示(邻接矩阵例子)
#include <iostream>
using namespace std;
//邻接矩阵存图:无权无相图

char v[105];
int g[105][105];
int n, m;
int vis[105];//标记是否被访问 v[x]=1 -->被访问过 想v[x]=0 -->未被访问

void vertex_build()
{
	for (int i = 0; i < n; i++)
	{
		cin >> v[i];
	}
	//为了方便直接存下标了(如果存入字符多一个find找下标的函数)
	int x, y;
	for (int i = 1; i <= m; i++)
	{
		cin >> x >> y;
		g[x][y] = g[y][x] = 1;//无向图传双向边
	}
}

void DFS(int x)
{
	//访问x
	cout << v[x] << "  ";
	vis[x] = 1;//标记x,说明x被访问
	for (int i = 0; i < n; i++)
	{
		if (g[x][i] == 1 && vis[i] == 0)
		{//v[i]没有被访问
			DFS(i);
		}
	}
}

int main()
{
	cin >> n >> m;
	vertex_build();

	for (int i = 0; i < n; i++)//防止非连通图
	{
		if (vis[i] == 0)
		{//说明i结点没有被访问,以i为起点进行DFS
			DFS(i);
		}
	}

	system("pause");
	return 0;
}
/*
9
15
ABCDEFGHI
0 1
0 5
1 6
5 6
2 1
1 8
2 8
6 7
2 3
3 8
3 7
3 4
4 7
4 5
3 6
*/
2.1.3算法效率分析

2.2广度优先遍历(BFS)
2.2.1算法思想

从图中某个顶点v出发,访问v,并置visited[v]的值为 true,然后将v进队。

只要队列不空,则重复下述过程

1.队头顶点u出队。

2.依次检查u的所有邻接点w,如果visited[w]的值为false,则访问w,并置visited[w]的值为true,然后将w进队。

2.2.2代码展示
#include <iostream>
using namespace std;
//邻接表存图:无权无相图
//BFS 可以求无权图中 起点到其他点的最短路径(层级就是路程)
int n, m;

//链表的结点结构
typedef struct EdgeNode
{
	int adj;//邻接点 可以是序号,可以就是本身数据,这里以下标为例
	struct EdgeNode* next;
}Edgenode;

//图的结点结构
struct veNode
{
	char data;
	EdgeNode* first;
};
veNode v[105];
int flag[105];//1为已入过队(为了标记这个结点已经被访问过)
int dist[105];//起点到i的距离
//-----队列
//循环队列
typedef struct
{
	int d[106];//结点s的数组
	int f, r;
}SQueue;

void queue_init(SQueue* q)
{
	q->f = q->r = 0;
}

void push_queue(SQueue* q, int x)
{
	if ((q->r + 1) % 106 == q->f)
	{
		cout << "队满" << endl;
		return;
	}
	q->d[q->r] = x;//初始的时候q->r是0,中括号里面的内容只是下标
	q->r = (q->r + 1) % 106;
}
int isEmpty(SQueue* q)
{
	if (q->f == q->r)
	{
		return 1;
	}
	else
	{
		return 0;
	}
}
int pop_queue(SQueue* q)
{
	if (isEmpty(q) == 1)
	{
		if ((q->r + 1) % 106 == q->f)
		{
			cout << "队空" << endl;
			return -1;
		}
	}
	int e = q->d[q->f];
	q->f = (q->f + 1) % 106;
	return e;
}
//-----
void vertex_build()
{
	for (int i = 0; i < n; i++)
	{
		cin >> v[i].data;
		v[i].first = NULL;
	}
	//为了方便直接存下标了(如果存入字符多一个find找下标的函数)
	int x, y;
	for (int i = 1; i <= m; i++)
	{
		cin >> x >> y;
		EdgeNode* s = new EdgeNode;
		s->adj = y;
		s->next = v[x].first;
		v[x].first = s;

		EdgeNode* s1 = new EdgeNode;
		s1->adj = x;
		s1->next = v[y].first;
		v[y].first = s1;
	}
}

void BFS(int x)//x是起点下标起点
{
	int e;//队首元素
	SQueue q;
	EdgeNode* p = NULL;
	queue_init(&q);

	//x(下标)入队
	push_queue(&q, x);
	flag[x] = 1;
	while (isEmpty(&q) == 0)
	{
		e = pop_queue(&q);
		cout << v[e].data << "  ";
		p = v[e].first;//队首元素pop出来后,找队首元素的first,看看这个结点后面有没有链表

		while (p != NULL)
		{
			if (flag[p->adj] == 0)
			{
				push_queue(&q, p->adj);
				dist[p->adj] = dist[e] + 1;
				flag[p->adj] = 1;
			}
			p = p->next;
		}
	}
}

int main()
{
	cin >> n >> m;
	vertex_build();


	for (int i = 0; i < n; i++)//防止非连通图
	{
		if (flag[i] == 0)
		{
			BFS(i);//将非连通图的起点下标传入
		}
	}
	cout << endl;
	for (int i = 0; i < n; i++)
	{
		cout << v[i].data << "  " << dist[i] << endl;
	}

	system("pause");
	return 0;
}
/*
9
15
ABCDEFGHI
0 1
0 5
1 6
5 6
2 1
1 8
2 8
6 7
2 3
3 8
3 7
3 4
4 7
4 5
3 6

*/
2.2.3算法效率分析

2.3总结

        图的遍历⽅式包括深度优先搜索(DFS)和⼴度优先搜索(BFS),其中 DFS 使⽤递归或栈进⾏ 实现,⽽ BFS 则采⽤队列进⾏实现。对⽐树的四种遍历⽅式,前序遍历、中序遍历和后序遍历均类似于 DFS,⽽层序遍历类似于 BFS,前中后序也均可采⽤栈的⽅式进⾏实现,层序遍历可以采⽤队列的⽅式 进⾏实现。

        这样看来,知识的融会贯通多么重要,总体⽽⾔,掌握下⾯的两条链,你便可以解决好多问题。

DFS → 前中后序 → 栈 → 线性表

BFS → 层序遍历 → 队列 → 链表

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1854795.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Typora + Hexo 图片路径问题(Typedown)

文章目录 1. 冲突来源2. 解决思路3. 实现1. typora图片路径2. hexo脚本 1. 冲突来源 Hexo上对于图片在md中的引用&#xff0c;使用了post_asset_folder: true配置&#xff0c;来更好的管理图片。 当一篇名为xxx.md的文章引用1.png图片时&#xff0c;默认让1.png保持在xxx文件夹…

【YOLO 系列】基于YOLO V8的车载摄像头交通信号灯检测识别系统【python源码+Pyqt5界面+数据集+训练代码】

前言 随着智能交通系统的发展&#xff0c;交通信号灯的准确识别对于提高道路安全和交通效率具有至关重要的作用。传统的交通信号灯识别方法依赖于固定的传感器和摄像头&#xff0c;存在安装成本高、维护困难等问题。为了解决这些问题&#xff0c;我们启动了这个项目&#xff0…

【2024.6.23】今日科技时事:科技前沿大事件

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

重复文件清理软件怎么用?分享3个删除重复文件的方法!

删除重复文件能够为电脑腾出很大的存储空间&#xff0c;不信&#xff1f;可以试试看哦&#xff01; 电脑使用久了&#xff0c;都会积累大量的文件&#xff0c;这其中难免会出现重复的文件&#xff0c;这些重复文件没有任何作用&#xff0c;而且会占用着电脑的空间&#xff0c;…

查找和排序

目录 一、查找 1.1查找的基本概念 1.2顺序查找 1.3折半查找&#xff08;二分查找&#xff09; 1.4散列表的查找 1.4.1基本概念 1.4.2散列函数的构造方法 1.4.3解决冲突的方法 二、排序 2.1排序的基本概念 2.2插入排序 2.2.1直接插入排序&#xff1a; 2.2.2希尔排序…

智能优化算法改进策略之局部搜索算子(三)—二次插值法

1、原理介绍 多项式是逼近函数的一种常用工具。在寻求函数极小点的区间&#xff08;即寻查区间&#xff09;上&#xff0c;我们可以利用在若干点处的函数值来构成低次插值多项式&#xff0c;用它作为求极小点的函数的近似表达式&#xff0c;并用这个多项式的极小点作为原函数极…

高可用电商支付架构设计方案

高可用电商支付架构设计 在现代电商业务中&#xff0c;支付过程是其中至关重要的一环&#xff0c;一个高可用、安全稳定的支付架构不仅可以提高整个系统的可靠性和扩展性&#xff0c;降低维护成本&#xff0c;还可以优化用户体验&#xff0c;增加用户黏性。 本文将提出一种高…

解析JSON字符串

QJsonDocument类用于解析JSON字符串&#xff0c;

学习C++第二天

1.缺省参数 缺省参数的概念&#xff1a; 缺省参数是声明或定义函数时为函数的参数指定一个缺省值。在调用该函数时&#xff0c;如果没有指定实参则采用该形参的缺省值&#xff0c;否则使用指定的实参。 void show(int a 10) {cout << a << endl; }int main() {sho…

英语智汇学习系统

目 录 1 软件概述 1.1 项目研究背景及意义 2 系统相关技术 2.1 HTML、WXSS、JAVASCRIPT技术 2.2 Vanilla框架 2.3 uni-app框架 2.4 MYSQL数据库 3 需求分析 3.1 可行性分析 3.2 功能需求分析 3.3 系统用户及用例分析 3.4 非功能需求分析 3.5 数据流图…

已解决ApplicationException异常的正确解决方法,亲测有效!!!

已解决ApplicationException异常的正确解决方法&#xff0c;亲测有效&#xff01;&#xff01;&#xff01; 目录 问题分析 出现问题的场景 报错原因 解决思路 解决方法 分析错误日志 检查业务逻辑 验证输入数据 确认服务器端资源的可用性 增加对特殊业务情况的处理…

android串口助手apk下载 源码 演示 支持android 4-14及以上

android串口助手apk下载 1、自动获取串口列表 2、打开串口就开始接收 3、收发 字符或16进制 4、默认发送at\r\n 5、android串口助手apk 支持android 4-14 &#xff08;Google seral port 太老&#xff09; 源码找我 需要 用adb root 再setenforce 0进入SELinux 模式 才有权限…

LabVIEW 控制 Tucsen 相机

LabVIEW 控制 Tucsen 相机 ucsen 是一家知名的显微镜相机制造商&#xff0c;其相机产品广泛应用于科研、工业和医疗等领域。本文将介绍如何使用 LabVIEW 软件来控制 Tucsen 相机&#xff0c;涵盖相机的基本情况、硬件和软件要求、具体的控制步骤和编程示例。通过使用 LabVIEW&…

【职场人】如何与同事有效沟通

在职场中&#xff0c;沟通如同桥梁&#xff0c;连接着每一位职场人士的心灵与智慧。有效的沟通不仅能让工作更加顺畅&#xff0c;还能让团队关系更加和谐。那么&#xff0c;如何与同事进行有效沟通呢&#xff1f;下面&#xff0c;我将结合个人经验和一些幽默的比喻&#xff0c;…

AI 大模型企业应用实战(11)-langchain 的Document Loader机制

loader机制让大模型具备实时学习的能力&#xff1a; 0 Loader机制 案例环境准备&#xff1a; import osos.environ["OPENAI_API_KEY"] "sk-javaedge" os.environ["OPENAI_PROXY"] "https://api.chatanywhere.tech"import os from …

开启调试模式

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 run()方法虽然适用于启动本地的开发服务器&#xff0c;但是每次修改代码后都要手动重启它。这样并不够方便&#xff0c;如果启用了调试支持&#xff…

[SAP ABAP] 删除内表数据

1.利用索引删除数据 语法格式 DELETE <itab> INDEX <idx>. <itab>&#xff1a;代表内表 <idx>&#xff1a;代表索引值 删除内表<itab>中的第<idx>条记录 示例1 lt_student内表中存在3条数据记录 我们使用如下指令删除内表中的第一条数…

WordPress强大多功能主题模板The7 v9.16.0

模板介绍 The7可以与WPBakery Page Builder&#xff08;原Visual Composer&#xff09;和Ultimate Addons的完全无缝集成。它也与大多数流行的插件完全兼容&#xff0c;例如WooCommerce&#xff0c;WPML&#xff0c;Yoast SEO&#xff0c;All in One WP Migration&#xff0c;…

解决MNIST数据集下载慢,或者Http连接失败问题

下载MNIST数据集时遇到速度慢的问题 解决&#xff1a;手动从MNIST数据集的官方网站直接使用下载好的数据文件&#xff0c;放到指定目录下&#xff0c;再进行调取即可。 手动下载地址&#xff1a;MNIST官网 http://yann.lecun.com/exdb/mnist/ 【仍需要连接外网】 这里我提供…

【大数据 复习】第3章 分布式文件系统HDFS(重中之重)

一、概念 1.分布式文件系统把文件分布存储到多个计算机节点上&#xff0c;通过网络实现、文件在多台主机上进行分布式存储的文件系统。&#xff08;就是你的电脑存a&#xff0c;我的电脑存pple&#xff09; 2.降低了硬件开销: 与之前使用多个处理器和专用高级硬件的并行化处理装…