深度优先(DFS)
理论基础
深度优先搜索(DFS, Depth-First Search)是图和树的遍历算法中的一种,它从一个节点开始,沿着树的边走到尽可能深的分支,直到节点没有子节点为止,然后回溯继续搜索下一个分支。DFS 是一种使用递归或栈实现的算法,它深入到每一个分支直到无路可走再退回到上一个分岔点,这种方式有助于解决许多搜索和路径问题。
DFS 的基本步骤
- 标记和访问:从一个未访问的节点开始,标记它为已访问。
- 递归探索:对于当前节点的每一个未访问的相邻节点,继续执行深度优先搜索(递归调用DFS)。
- 回溯:当一个节点的所有相邻节点都被访问后,回溯到之前的节点继续探索未访问的节点。
DFS 的特点
- 深度优先:它尝试尽可能深地搜索图的分支。
- 使用栈:无论是显式使用栈还是通过递归调用的隐式栈,DFS 都利用栈先进后出的特性来管理节点和回溯。
- 可能非最短路径:DFS 不保证找到的是最短路径,它可能在找到目标前遍历图中大部分节点。
- 解空间树:在涉及路径和状态的问题中,DFS 常用于生成解空间树,并通过回溯剪枝。
- 复杂性:在最坏的情况下,DFS 的时间复杂度是 O(V+E),其中 V 是顶点数,E 是边数。
DFS 的应用场景
- 路径搜索:在迷宫或图中查找从起点到终点的路径。
- 拓扑排序:在有向无环图中进行拓扑排序。
- 连通分量:在无向图中查找所有连通分量。
- 解决问题:如解决迷宫问题、路径问题和那些可以通过回溯方法解决的问题。
简单总结
深度优先搜索是一个递归的过程,它尝试深入到每一个分支直到不能再深入为止,然后通过回溯继续探索其他分支。它的核心在于尝试所有可能的路径直到找到解决方案或覆盖所有的节点。DFS 的记忆点在于它的递归性质和回溯机制,这使得它非常适合处理需要探索所有可能性的问题。
前提
JAVA实验环境
实现Bag抽象数据结构
实现Stack抽象数据结构
实现无向图的构建
数据结构
本算发中涉及到的基本数据结构
private boolean[] marked
private int[] edgeTo
private final int s
myLinkedStack //一种栈的实现,具体见以下链接
myGraph //一种无向图的实现
myGraph的数据结构图
实验数据和实验环境
本实验要求根据DFS算法实现从任意一点的深度优先路径的打印功能
实验数据是一个名为tinyCG.txt的文件,用来构建如下图所示的无向图
算法流程
根据以上深度优先搜索得到的结果,整理输出从起点到图中所有点的
下图为从0到5的深度优先路劲的计算过程,我们可以计算出从任意起点到其他所有点的深度优先路径
代码实现
import edu.princeton.cs.algs4.In;
import edu.princeton.cs.algs4.StdOut;
public class myDepthFirstPaths {
private boolean[] marked;
private int[] edgeTo;
private final int s;
public myDepthFirstPaths(myGraph G, int s){
this.s = s;
marked = new boolean[G.V()];
edgeTo = new int[G.V()];
dfs(G, s);
}
private void dfs