图的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的更多相关文章

  1. 图的DFS和BFS(邻接表)

    用C++实现图的DFS和BFS(邻接表) 概述   图的储存方式有邻接矩阵和邻接表储存两种.由于邻接表的实现需要用到抽象数据结构里的链表,故稍微麻烦一些.C++自带的STL可以方便的实现List,使算 ...

  2. Java数据结构——图的DFS和BFS

    1.图的DFS: 即Breadth First Search,深度优先搜索是从起始顶点开始,递归访问其所有邻近节点,比如A节点是其第一个邻近节点,而B节点又是A的一个邻近节点,则DFS访问A节点后再访 ...

  3. 数据结构(11) -- 邻接表存储图的DFS和BFS

    /////////////////////////////////////////////////////////////// //图的邻接表表示法以及DFS和BFS //////////////// ...

  4. [数据结构]图的DFS和BFS的两种实现方式

    深度优先搜索 深度优先搜索,我们以无向图为例. 图的深度优先搜索(Depth First Search),和树的先序遍历比较类似. 它的思想:假设初始状态是图中所有顶点均未被访问,则从某个顶点v出发, ...

  5. 图的DFS与BFS遍历

    一.图的基本概念 1.邻接点:对于无向图无v1 与v2之间有一条弧,则称v1与v2互为邻接点:对于有向图而言<v1,v2>代表有一条从v1到v2的弧,则称v2为v1的邻接点. 2.度:就是 ...

  6. 树的常见算法&图的DFS和BFS

    树及二叉树: 树:(数据结构中常见的树) 树的定义

  7. 图、dfs、bfs

    graphdfsbfs 1.clone graph2.copy list with random pointer3.topological sorting4.permutations5.subsets ...

  8. 树与图的DFS与BFS

    树的DFS 题目:https://www.acwing.com/problem/content/848/ 代码 #include<bits/stdc++.h> using namespac ...

  9. 邻接矩阵实现图的存储,DFS,BFS遍历

    图的遍历一般由两者方式:深度优先搜索(DFS),广度优先搜索(BFS),深度优先就是先访问完最深层次的数据元素,而BFS其实就是层次遍历,每一层每一层的遍历. 1.深度优先搜索(DFS) 我一贯习惯有 ...

随机推荐

  1. 为什么SpringBoot项目里引入其他依赖不要写版本号

    <dependencies> <dependency> <groupId>org.springframework.boot</groupId> < ...

  2. java消除整型数组中重复的元素,排序后输出新数组

    法一: import java.util.Arrays; import java.util.Scanner; public class Main { public static void main(S ...

  3. 2Ants(独立,一个个判,弹性碰撞,想象)

    AntsDescriptionAn army of ants walk on a horizontal pole of length l cm, each with a constant speed ...

  4. 《JavaScript语言入门教程》记录整理:入门和数据类型

    目录 入门篇 js介绍 历史 基本语法 数据类型 概述 null 和 undefined 数值 字符串 对象 函数 数组 本系列基于阮一峰老师的<JavaScrip语言入门教程>或< ...

  5. PHP zip_entry_filesize() 函数

    定义和用法 The zip_entry_filesize() 函数返回 zip 档案项目的原始文件尺寸(在压缩之前).高佣联盟 www.cgewang.com 语法 zip_entry_filesiz ...

  6. MOSFET 的 I / V 特性曲线

    https://www.cnblogs.com/yeungchie/ MOSFET 线性区(三极管区,\(V_{DS} \leq V_{GS} - V_{TH}\)) \[I_{D} = \mu_{n ...

  7. Scala---初探

    scala语言量大特性:面向对象+函数式编程 Scala的类型 val指的是引用不可变,而不是值. 值类型 Byte Char Short Int Long Float Double 引用类型 Str ...

  8. lamp架构搭建

    目录 1. LAMP架构介绍 2.web服务器工作流程 2.1 cgi与fastcgi 2.2 httpd与php结合的方式 2.3 web工作流程 3. lamp平台搭建 3.1 安装httpd 3 ...

  9. 学习java 线程池-1: ThreadPoolExecutor

    1. Executor 该接口内只有一个接口方法 :该方法的目的就是执行指定的 Runnable (但会不会执行,或者会不会立马执行,则不一定.因为要取决于整个线程池的状态) Executor 中文的 ...

  10. 【AHOI2009】中国象棋 题解(线性DP+数学)

    前言:这题主要是要会设状态,状态找对了问题迎刃而解. --------------------------- 题目描述 这次小可可想解决的难题和中国象棋有关,在一个N行M列的棋盘上,让你放若干个炮(可 ...