图的DFS与BFS
图的DFS与BFS(C++)
概述
大一学生,作为我的第一篇Blog,准备记录一下图的基本操作:图的创建与遍历。请大佬多多包涵勿喷。
图可以采用邻接表,邻接矩阵,十字链表等多种储存结构进行储存,这里为了方便演示算法,采用邻接矩阵。 图为边的权值都默认为1的无向图 。
国内大学现行教材大多是C语言实现,然而C语言其实并不适合实现ADT,故这里使用和C语言相近的C++,引入OOP的机制进行类封装,更直观和容易理解。
代码
1.首先,因为要用到DFS,这里分别采用递归的方式和非递归方式(要用到STL提供的栈),故引入头文件,BFS利用队列,引用头文件queue。
接着对图的抽象数据类型进行声明,类的属性有两个,分别是储存顶点容器vertex和储存边的容器edge,提供3个接口,分别对应三种遍历方式。
#include<iostream>
#include<queue>
#include<vector>
#include<stack>
#include<mem.h>
int inf=-9999;
using namespace std;
//为了增强复用性,这里封装成类模板,虚拟类型为T
template <class T>
struct Graph{
//存放顶点
vector<T> vertex;
//存放边
vector<vector<int>> edge;
//标记数组
bool book[100];
//构造函数
Graph(int n,int m);
//析构函数
~Graph();
//递归深度优先遍历
void DFS_recursion(int cur);
//非递归深度优先遍历
void DFS_stack(int cur);
//广度优先遍历
void BFS(int cur);
};
2.接着,定义Graph类的构造函数与析构函数,构造函数传入两个参数,对应图的定点数和边数,并创建一个book标记数组来记录定点是否访顶点,初始化book全部置为0,表示顶点都没访问。还有一点值得注意的是:这里为了方便,直接将顶点的数值设置为顶点的下标。
template <class T>
Graph<T>::Graph(int n,int m){
//为了使顶点对应的下标符合人类思维(从1起),这里的容器大小分配为n+1;
vertex.resize(n+1);
edge.resize(n+1);
for(int i=0;i<n+1;i++){
edge[i].resize(n+1);
}
//初始化邻接矩阵
for(int i=1;i<=n;i++){
for(int j=1;j<=n;j++){
if(i==j) edge[i][j]=0;
else edge[i][j]=inf;
}
}
cout<<"请输入各条边的两个邻点"<<endl;
for(int i=1;i<=m;i++){
int a,b;
cin>>a>>b;
//这里使用无向图,故边应用edge[a][b]=edge[b][a]=1;初始化
edge[a][b]=edge[b][a]=1;
}
////因为各顶点的值为设为int类型,故为了方便,将各顶点的值初始化为其下标
for(int i=1;i<=m;i++){
vertex[i]=i;
}
//标记数组初始化
memset(book,0,100);
}
template <class T>
Graph<T>:: ~Graph(){
vertex.clear();
edge.clear();
}
3.以下分别是3种不同的图的遍历方法的实现:
template <class T>
void Graph<T>::DFS_recursion(int cur){
book[cur]=true;
cout<<vertex[cur];
for(int i=0;i<vertex.size();i++){
if(edge[cur][i]==1&&book[i]==0){
DFS_recursion(i);
}
}
}
template <class T>
void Graph<T>::DFS_stack(int cur){
stack<int> s;
s.push(cur);
while(!s.empty()){
int top=s.top();
if(book[top]==0){
cout<<top;
book[top]=1;
}
else s.pop();
for(int i=0;i<vertex.size();i++){
if(book[i]==0&&edge[top][i]==1){
s.push(i);
break;
}
}
}
return;
}
template <class T>
void Graph<T>::BFS(int cur){
queue<int> q;
q.push(cur);
book[cur]=1;
while(!q.empty()){
int front=q.front();
q.pop();
cout<<vertex[front];
for(int i=0;i<vertex.size();i++){
if(edge[front][i]==1&&book[i]==0){
q.push(i);
book[i]=1;
}
}
}
}
4.最后是测试部分:主函数创建实例,并调用类的方法完成对图的操作。这里初始化为5个节点5条边,权值都默认为1。调用时都从顶点1开始便遍历。注意:每次便利完成后都要用memset将book标记数组全部置0,表示没访问过。
int main(){
//将类模板实例化为模板类,再将模板类实例化为一个对象
Graph<T>* G=new Graph<int>(5,5);
cout<<"深度优先遍历(递归)如下:"<<endl;
G->DFS_recursion(1);
memset(G->book,0,100);
cout<<endl;
cout<<"深度优先遍历(非递归)如下:"<<endl;
G->DFS_stack(1);
memset(G->book,0,100);
cout<<endl;
cout<<"广度优先队列遍历如下:"<<endl;
G->BFS(1);
cout<<endl;
system("pause");
return 0;
}
输出结果
测试控制台输出如下:

图的DFS与BFS的更多相关文章
- 图的DFS和BFS(邻接表)
用C++实现图的DFS和BFS(邻接表) 概述 图的储存方式有邻接矩阵和邻接表储存两种.由于邻接表的实现需要用到抽象数据结构里的链表,故稍微麻烦一些.C++自带的STL可以方便的实现List,使算 ...
- Java数据结构——图的DFS和BFS
1.图的DFS: 即Breadth First Search,深度优先搜索是从起始顶点开始,递归访问其所有邻近节点,比如A节点是其第一个邻近节点,而B节点又是A的一个邻近节点,则DFS访问A节点后再访 ...
- 数据结构(11) -- 邻接表存储图的DFS和BFS
/////////////////////////////////////////////////////////////// //图的邻接表表示法以及DFS和BFS //////////////// ...
- [数据结构]图的DFS和BFS的两种实现方式
深度优先搜索 深度优先搜索,我们以无向图为例. 图的深度优先搜索(Depth First Search),和树的先序遍历比较类似. 它的思想:假设初始状态是图中所有顶点均未被访问,则从某个顶点v出发, ...
- 图的DFS与BFS遍历
一.图的基本概念 1.邻接点:对于无向图无v1 与v2之间有一条弧,则称v1与v2互为邻接点:对于有向图而言<v1,v2>代表有一条从v1到v2的弧,则称v2为v1的邻接点. 2.度:就是 ...
- 树的常见算法&图的DFS和BFS
树及二叉树: 树:(数据结构中常见的树) 树的定义
- 图、dfs、bfs
graphdfsbfs 1.clone graph2.copy list with random pointer3.topological sorting4.permutations5.subsets ...
- 树与图的DFS与BFS
树的DFS 题目:https://www.acwing.com/problem/content/848/ 代码 #include<bits/stdc++.h> using namespac ...
- 邻接矩阵实现图的存储,DFS,BFS遍历
图的遍历一般由两者方式:深度优先搜索(DFS),广度优先搜索(BFS),深度优先就是先访问完最深层次的数据元素,而BFS其实就是层次遍历,每一层每一层的遍历. 1.深度优先搜索(DFS) 我一贯习惯有 ...
随机推荐
- 题解 CF1385D 【a-Good String】
题意 定义:字符串s 为一个c-好串(c 为一个字符)时,必须满足: 当\(|s| = 1\) ,\(s = c\) 当\(|s| > 1\), \(s\) 的左半部分为全为 \(c\),右半部 ...
- Jmeter(十八) - 从入门到精通 - JMeter后置处理器 -下篇(详解教程)
1.简介 后置处理器是在发出“取样器请求”之后执行一些操作.取样器用来模拟用户请求,有时候服务器的响应数据在后续请求中需要用到,我们的势必要对这些响应数据进行处理,后置处理器就是来完成这项工作的.例如 ...
- xilinx fpga 生成3*3窗口
在写滤波程序的时候在网上看了好几篇大佬的笔记,都有提到使用3*3窗口,由于小白一个,看到复杂的理论就惧怕的不行.但是现在不得不上,自己调用移位寄存器ip核然后做了个3*3窗口出来,自己动手作出来忽然感 ...
- python-scrapy爬虫框架爬取拉勾网招聘信息
本文实例为爬取拉勾网上的python相关的职位信息, 这些信息在职位详情页上, 如职位名, 薪资, 公司名等等. 分析思路 分析查询结果页 在拉勾网搜索框中搜索'python'关键字, 在浏览器地址栏 ...
- Python编程之美:最佳实践指南PDF高清完整版免费下载|百度云盘|Python新手到进阶
百度云盘:Python编程之美:最佳实践指南PDF高清完整版免费下载 提取码:1py6 内容简介 <Python编程之美:最佳实践指南>是Python用户的一本百科式学习指南,由Pytho ...
- Image Processing Using Multi-Code GAN Prior, CVPR2020
论文:Image Processing Using Multi-Code GAN Prior, CVPR2020 代码:https://github.com/genforce/mganprior 这是 ...
- SQL语句中带有EXISTS谓词的子查询的理解与使用
EXISTS:代表存在量词. 在SQL中,把具有全称量词的谓词查询问题转换成等价的存在量词的谓词查询予以实现. 如有三个表,Student(Sno,Sname),Course(Cno,Cname),S ...
- OpenLDAP on Centos7
一.环境准备 echo nameserver 114.114.114.114 > /etc/resolv.conf ##更改DNSecho 192.168.0.190 hello.com > ...
- Python网络编程基础 PDF 完整超清版|网盘链接内附提取码下载|
点此获取下载地址提取码:y9u5 Python网络编程最好新手入门书籍!175个详细案例,事实胜于雄辩,Sockets.DNS.Web Service.FTP.Email.SMTP.POP.IMAP. ...
- TCP 服务器端
""" 建立tcp服务器 绑定本地服务器信息(ip地址,端口号) 进行监听 获取监听数据(监听到的客户端和地址) 使用监听到的客户端client_socket获取数据 输 ...