图的深度优先遍历(DFS)和广度优先遍历(BFS)
body, table{font-family: 微软雅黑; font-size: 13.5pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}
右图是递归遍历的过程,其实每一层都是从A结点开始搜寻满足条件的点 |
|
/* DFS.h */
#ifndef __DFS_H__
#define __DFS_H__
#include"Graph.h"
namespace meihao
{
void DFS(const meihao::Graph& g,int vi,bool*& visited); //参数->图和顶点数组中某个顶点的下标
void DFSTraversal(const meihao::Graph& g);
};
#endif
/* testMain.cpp */ #include"DFS.h"
#include<iostream>
int main()
{
meihao::Graph g("data.txt");
meihao::DFSTraversal(g);
cout<<endl;
system("pause");
}
/* data.txt */ 9
A B C D E F G H I
0 1 0 0 0 1 0 0 0
1 0 1 0 0 0 1 0 1
0 1 0 1 0 0 0 0 1
0 0 1 0 1 0 1 1 1
0 0 0 1 0 1 0 1 0
1 0 0 0 1 0 1 0 0
0 1 0 1 0 1 0 1 0
0 0 0 1 1 0 1 0 0
0 1 1 1 0 0 0 0 0
|
/* DFS.cpp */
#include"DFS.h"
namespace meihao
{
//算法都是基于邻接矩阵实现的
void DFS(const meihao::Graph& g,int vi,bool*& visited)
{
visited[vi] = true; //修改第vi个结点的访问标记为true
cout<<g.getGraphVertexData(vi)<<" ";
for(int idx=0;idx!=g.getGraphVertexNumber();++idx)
{
if(1==g.getGraphEdgeWeight(vi,idx)&&
false==visited[idx]) //如果(vi,idx)之间存在边(==1),并且第idx个顶点还没有访问过
{
DFS(g,idx,visited); //递归遍历第idx个顶点
}
}
}
void DFSTraversal(const meihao::Graph& g)
{
bool* visited = new bool[g.getGraphVertexNumber()]();
for(int idx=0;idx!=g.getGraphVertexNumber();++idx)
{
visited[idx] = false; //初始化访问标记,全部为false,表示未访问
}
for(int idx=0;idx!=g.getGraphVertexNumber();++idx)
{
if(false==visited[idx]) //随便选一个点,如果未访问过,就从它开始深度优先遍历
DFS(g,idx,visited);
}
}
};
|
/* BFS.h */
#ifndef __BFS_H__
#define __BFS_H__
#include"Graph.h"
namespace meihao
{
void BFSTraversal(const meihao::Graph& g);
};
#endif
/* test.cpp */ #include"BFS.h"
#include<iostream>
int main()
{
meihao::Graph g("data.txt");
meihao::BFSTraversal(g);
cout<<endl;
system("pause");
}
/* data.txt */
9
A B C D E F G H I
0 1 0 0 0 1 0 0 0
1 0 1 0 0 0 1 0 1
0 1 0 1 0 0 0 0 1
0 0 1 0 1 0 1 1 1
0 0 0 1 0 1 0 1 0
1 0 0 0 1 0 1 0 0
0 1 0 1 0 1 0 1 0
0 0 0 1 1 0 1 0 0
0 1 1 1 0 0 0 0 0
|
/* BFS.cpp */
#include"BFS.h"
#include<queue>
namespace meihao
{
void BFSTraversal(const meihao::Graph& g)
{ //广度优先遍历相当于层序遍历
queue<int> rootNode; //存放图的顶点
bool* visited = new bool[g.getGraphVertexNumber()];
for(int idx=0;idx!=g.getGraphVertexNumber();++idx)
{
visited[idx] = false;
}
for(int idx=0;idx!=g.getGraphVertexNumber();++idx)
{ //if语句可以确保如果图中有多个连通分量,也能每个点都访问到
if(false==visited[idx]) //如果该结点没有访问到
{
//访问
cout<<g.getGraphVertexData(idx)<<" ";
visited[idx] = true;
rootNode.push(idx);
while(!rootNode.empty()) //把刚刚访问到的结点的下一层结点访问并入队列
{
for(int iidx=0;iidx!=g.getGraphVertexNumber();++iidx)
{
if(1==g.getGraphEdgeWeight(rootNode.front(),iidx)&&
false==visited[iidx])
{
cout<<g.getGraphVertexData(iidx)<<" ";
visited[iidx] = true;
rootNode.push(iidx);
}
}
rootNode.pop(); //最先访问的一个结点出队列
}
}
}
}
};
|
图的深度优先遍历(DFS)和广度优先遍历(BFS)的更多相关文章
- 图的深度优先搜索(DFS)和广度优先搜索(BFS)算法
深度优先(DFS) 深度优先遍历,从初始访问结点出发,我们知道初始访问结点可能有多个邻接结点,深度优先遍历的策略就是首先访问第一个邻接结点,然后再以这个被访问的邻接结点作为初始结点,访问它的第一个邻接 ...
- 深度优先搜索DFS和广度优先搜索BFS简单解析(新手向)
深度优先搜索DFS和广度优先搜索BFS简单解析 与树的遍历类似,图的遍历要求从某一点出发,每个点仅被访问一次,这个过程就是图的遍历.图的遍历常用的有深度优先搜索和广度优先搜索,这两者对于有向图和无向图 ...
- 深度优先搜索DFS和广度优先搜索BFS简单解析
转自:https://www.cnblogs.com/FZfangzheng/p/8529132.html 深度优先搜索DFS和广度优先搜索BFS简单解析 与树的遍历类似,图的遍历要求从某一点出发,每 ...
- 图的 储存 深度优先(DFS)广度优先(BFS)遍历
图遍历的概念: 从图中某顶点出发访遍图中每个顶点,且每个顶点仅访问一次,此过程称为图的遍历(Traversing Graph).图的遍历算法是求解图的连通性问题.拓扑排序和求关键路径等算法的基础.图的 ...
- 图的深度优先遍历(DFS)和广度优先遍历(BFS)算法分析
1. 深度优先遍历 深度优先遍历(Depth First Search)的主要思想是: 1.首先以一个未被访问过的顶点作为起始顶点,沿当前顶点的边走到未访问过的顶点: 2.当没有未访问过的顶点时,则回 ...
- 深度优先搜索DFS和广度优先搜索BFS
DFS简介 深度优先搜索,一般会设置一个数组visited记录每个顶点的访问状态,初始状态图中所有顶点均未被访问,从某个未被访问过的顶点开始按照某个原则一直往深处访问,访问的过程中随时更新数组visi ...
- 图的遍历(搜索)算法(深度优先算法DFS和广度优先算法BFS)
图的遍历的定义: 从图的某个顶点出发访问遍图中所有顶点,且每个顶点仅被访问一次.(连通图与非连通图) 深度优先遍历(DFS): 1.访问指定的起始顶点: 2.若当前访问的顶点的邻接顶点有未被访问的,则 ...
- 【C++】基于邻接矩阵的图的深度优先遍历(DFS)和广度优先遍历(BFS)
写在前面:本博客为本人原创,严禁任何形式的转载!本博客只允许放在博客园(.cnblogs.com),如果您在其他网站看到这篇博文,请通过下面这个唯一的合法链接转到原文! 本博客全网唯一合法URL:ht ...
- 图的深度优先搜索dfs
图的深度优先搜索: 1.将最初访问的顶点压入栈: 2.只要栈中仍有顶点,就循环进行下述操作: (1)访问栈顶部的顶点u: (2)从当前访问的顶点u 移动至顶点v 时,将v 压入栈.如果当前顶点u 不存 ...
随机推荐
- 字符串、字节数组、流之间的相互转换以及文件MD5的计算
using System; using System.IO; using System.Security.Cryptography; using System.Text; namespace myMe ...
- 雷林鹏分享:jQuery EasyUI 表单 - 过滤下拉数据网格
jQuery EasyUI 表单 - 过滤下拉数据网格 下拉数据网格(Combogrid)组件和下拉框(Combobox)组件的共同点是,除了都具有下拉面板以外,它们都是基于数据网格(Datagrid ...
- Python中字典和集合的用法
本人开始学习python 希望能够慢慢的记录下去 写下来只是为了害怕自己忘记. python中的字典和其他语言一样 也是key-value的形式 利用空间换时间 可以进行快速的查找 key 是唯一的 ...
- vue 基础(一)
一 vue.js的M-V-VM思想 MVVM 是Model-View-ViewModel 的缩写,它是一种基于前端开发的架构模式. 1.Model指代的就是vue对象的data属性里面的数据.这里的数 ...
- 如何在mmseg3添加词库
一.了解几个文件 ll /usr/local/mmseg/etc/mmseg.ini uni.lib 编译后的词库,给sphinx 用的unigram.txt 原词库给人看的, 在这里面人工添加自 ...
- python记录_day33 线程
##进程就像加工厂,线程是里边的流水线##进程是资源单位,线程是运行单位,每个进程至少有一个线程 即进程是资源分配的最小单位,线程是CPU调度的最小单位 一.线程的创建两种方式,和进程类似1.t = ...
- loj#6491. zrq 学反演
题意:求\(\sum_{i_1=1}^m\sum_{i_2=1}^m...\sum_{i_n=1}^mgcd(i_1,i_2,...i_n)\) 题解:\(\sum_{d=1}^md\sum_{i_1 ...
- verilog的移位运算符(存在不公平现象)
从上面的例子可以看出,start在移过两位以后,用0来填补空出的位.进行移位运算时应注意移位前后变量的位数,下面举例说明. 4’b1001<<1 = 5’b10010; //左移1位后用0 ...
- 2017-4-28/PHP实现Redis
谈一谈Redis的数据结构,如果换做PHP,怎么实现?如果再考虑用上LFU或LRU,又该如何实现? Redis的数据结构有String.List.Set.Sorted Set.Hash等,而PHP ...
- JavaScript中 null 的 typeof是object
JavaScript中 null 的 typeof是object