一、图的基本概念

1、邻接点:对于无向图无v1 与v2之间有一条弧,则称v1与v2互为邻接点;对于有向图而言<v1,v2>代表有一条从v1到v2的弧,则称v2为v1邻接点。

2、度:就是与该顶点相互关联的弧的个数。

3、连通图:无向图的每个顶点之间都有可达路径,则称该无向图为连通图。有向图每个顶点之间都有<v1,v2>和<v2,v1>,则称此有向图为强连通图

二、存储结构

1、邻接矩阵存储(Adjacency Matrix)

对无权图,顶点之间有弧标1,无弧标0;

对有权图,顶点之间有弧标权,无弧标无穷大;

2、邻接表(看书吧,精力有限不想画图了)

三、DFS与BFS

其实深度优先和广度优先,最重要的是要掌握其思想,而不是具体的实现,因为万变不离其宗。

1、DFS深度优先

①从v0出发访问v0

②找到刚访问过得顶点,访问其未访问过得邻接点中的一个。重复②直到不再有邻接点位置。

③回溯,返回前一个被访问过但是仍有未被访问过得邻接点的顶点,继续访问它的下一个邻接点。重复②直至完全遍历。

可能我描述的不够准确,但那也没有办法,应为有些东西真的是只能意会。学计算机必须要结合具体的例子来看。

原图:

DFS

其实简单来说:就是如果访问到的结点有邻接点就一直向下访问,否则就回溯。同树中的先根遍历很类似。

数据存储结构:

String[] vertex = new String[] {"v0","v1","v2","v3","v4","v5"};
int[][] matrix = new int[][] {
{0,1,0,1,0,0},
{1,0,1,0,1,0},
{0,1,0,0,0,0},
{1,0,0,0,0,1},
{0,1,0,0,0,0},
{0,0,0,1,0,0}
};
int[] visited = new int[6];//标记访问过得结点

递归实现:

void matrixDFS(int v0) {
System.out.print(vertex[v0] + " ");
visited[v0] = 1;
//遍历寻找v0的邻接点
for(int i = 0; i < vertex.length; i++) {
if(matrix[v0][i] == 1 && visited[i] == 0) {
matrixDFS(i);
}
}
}

非递归实现:

//基于栈的非递归实现
void matrixDFS1(int v0) {
Stack<Integer> stack = new Stack<Integer>();
stack.push(v0);
while(!stack.isEmpty()) {
int v = stack.pop();
if(visited[v] == 0) {//栈中很可能有重复的元素
System.out.print(vertex[v] + " ");
visited[v] = 1;
for(int i = 0; i < matrix.length; i++) {
if(matrix[v][i] == 1 && visited[i] == 0) {
stack.push(i);
}
}
}
}
}

 2、BFS深度优先遍历

①访问v0

②访问v0所以未被访问过的邻接点

③分别以这些邻接点(②中的邻接点)出发,去访问他们的邻接点。重复③知道遍历结束。

BFS就比较像树中的层次遍历

代码实现:

void matrixBFS2(int v0) {
Queue<Integer> queue = new LinkedList<>();
queue.offer(v0);
while(!queue.isEmpty()) {
int v = queue.poll();
if(visited[v] == 0) {
System.out.print(vertex[v] + " ");
visited[v] = 1;
for(int i = 0; i < vertex.length; i++) {
if(matrix[v][i] == 1 && visited[i] == 0) {
queue.offer(i);
}
}
}
}
}

可以看到这段代码和DFS递归实现的代码差别只是一个用栈一个用队列。

DFS:访问v0,邻接点如栈

BFS:访问v0,邻接点入队

我们的目标不是要掌握这些具体的实现,而是理解DFS与BFS算法的思想,将其活学活用。

数据结构与算法参考

最新文章

  1. PowerShell vs. PsExec for Remote Command Execution
  2. [译]JavaScript:将字符串两边的双引号转换成单引号
  3. 使用GoodFeaturesToTrack进行关键点检测---29
  4. web.xml基本配置描述
  5. nodejs问题整理--fs.exists无法正确判断文件的问题
  6. Map的迭代
  7. 【京东详情页】——原生js学习之匿名函数
  8. 十一.keepalived高可用服务实践部署
  9. 简单快速的Android打渠道包的方法
  10. UVA11324 The Largest Clique (强连通缩点+DP最长路)
  11. 05Hadoop-左外连接
  12. office 安装
  13. GTest的安装与使用
  14. 用最通俗的话解释AJAX是什么东西
  15. Java 中转换为String类型的四种方法
  16. 洛谷 P5206: bzoj 5475: LOJ 2983: [WC2019] 数树
  17. git 添加review的相关操作
  18. DAY2-Flask项目
  19. c/c++ int long float double 表示范围
  20. centos7 下pycharm无法输入中文问题解决方案

热门文章

  1. 【CUDA 基础】5.0 共享内存和常量内存
  2. Job for docker.service failed because the control process exited with error code. See
  3. Eratos筛法(筛选素数)
  4. Python 生成随机数函数和加密函数(MD5)
  5. R语言:各类型数据文件的导入
  6. [go]包管理
  7. Android下文件访问的权限
  8. rally使用tempest进行测试
  9. PHP md5() 函数
  10. RabbitMQ学习之:(八)Topic Exchange (转贴+我的评论)