图论01-DFS和BFS(深搜和广搜邻接矩阵和邻接表/Java)

1.深度优先理论基础(dfs)

  • dfs的两个关键操作
java 复制代码
搜索方向,是认准一个方向搜,直到碰壁之后再换方向
换方向是撤销原路径,改为节点链接的下一个路径,回溯的过程。
  • dfs解题模板
java 复制代码
void dfs(参数) {
    if (终止条件) {
        存放结果;
        return;
    }

    for (选择:本节点所连接的其他节点) {
        处理节点;
        dfs(图,选择的节点); // 递归
        回溯,撤销处理结果
    }
}
  • Java代码实现

邻接矩阵表示的图

java 复制代码
public class DFSTraversalRecursive {
    private int[][] adjacencyMatrix; // 邻接矩阵
    private boolean[] visited; // 用于标记节点是否被访问过
    private int numNodes; // 节点数量

    public DFSTraversalRecursive(int[][] matrix) {
        this.adjacencyMatrix = matrix;
        this.numNodes = matrix.length;
        this.visited = new boolean[numNodes];
    }

    // 递归实现的深度优先搜索遍历
    public void dfsTraversalRecursive(int startNode) {
        visited[startNode] = true; // 标记当前节点为已访问
        System.out.print(startNode + " "); // 输出当前节点

        for (int i = 0; i < numNodes; i++) {
            // 如果存在从当前节点到节点 i 的边,并且节点 i 还未被访问过
            if (adjacencyMatrix[startNode][i] == 1 && !visited[i]) {
                dfsTraversalRecursive(i); // 递归调用,从节点 i 开始深度优先搜索
            }
        }
    }
}

邻接表表示的图

java 复制代码
public class DFSTraversalAdjacencyList {
    private List<List<Integer>> adjacencyList; // 邻接表存储图的结构
    private boolean[] visited; // 标记节点是否被访问过

    // 构造函数,初始化邻接表和visited数组
    public DFSTraversalAdjacencyList(int numNodes) {
        this.adjacencyList = new ArrayList<>();
        for (int i = 0; i < numNodes; i++) {
            this.adjacencyList.add(new ArrayList<>());
        }
        this.visited = new boolean[numNodes];
    }

    // 添加边到邻接表
    public void addEdge(int source, int destination) {
        adjacencyList.get(source).add(destination);
    }

    // 递归实现的深度优先搜索遍历
    public void dfsTraversalRecursive(int startNode) {
        visited[startNode] = true; // 标记当前节点为已访问
        System.out.print(startNode + " "); // 输出当前节点

        // 遍历当前节点的所有邻居节点
        for (int neighbor : adjacencyList.get(startNode)) {
            if (!visited[neighbor]) {
                dfsTraversalRecursive(neighbor); // 递归调用,从邻居节点开始深度优先搜索
            }
        }
    }

2.广度优先搜索理论基础(bfs)

  • 使用场景
java 复制代码
广搜的搜索方式就适合于解决两个点之间的最短路径问题。

因为广搜是从起点出发,以起始点为中心一圈一圈进行搜索,一旦遇到终点,记录之前走过的节点就是一条最短路
  • Java代码实现

邻接矩阵表示的图

java 复制代码
public class BFSTraversalAdjacencyMatrix {
    private int[][] adjacencyMatrix; // 邻接矩阵存储图的结构
    private boolean[] visited; // 标记节点是否被访问过

    public BFSTraversalAdjacencyMatrix(int numNodes) {
        this.adjacencyMatrix = new int[numNodes][numNodes]; // 初始化邻接矩阵
        this.visited = new boolean[numNodes]; // 初始化visited数组
    }

    // 添加边到邻接矩阵
    public void addEdge(int source, int destination) {
        adjacencyMatrix[source][destination] = 1;
    }

    // 广度优先搜索遍历
    public void bfsTraversal(int startNode) {
        Queue<Integer> queue = new LinkedList<>(); // 创建一个队列用于BFS遍历
        queue.add(startNode); // 将起始节点加入队列
        visited[startNode] = true; // 标记起始节点为已访问

        while (!queue.isEmpty()) {
            int currentNode = queue.poll(); // 出队列一个节点
            System.out.print(currentNode + " "); // 输出当前节点

            for (int i = 0; i < adjacencyMatrix.length; i++) {
                if (adjacencyMatrix[currentNode][i] == 1 && !visited[i]) {
                    queue.add(i); // 将未访问的邻居节点加入队列
                    visited[i] = true; // 标记邻居节点为已访问
                }
            }
        }
    }
}

邻接表表示的图

java 复制代码
public class BFSTraversalAdjacencyList {
    private LinkedList<Integer>[] adjacencyList; // 邻接表存储图的结构
    private boolean[] visited; // 标记节点是否被访问过

    public BFSTraversalAdjacencyList(int numNodes) {
        // 初始化邻接表
        this.adjacencyList = new LinkedList[numNodes];
        for (int i = 0; i < numNodes; i++) {
            adjacencyList[i] = new LinkedList<Integer>();
        }
        // 初始化visited数组
        this.visited = new boolean[numNodes];
    }

    // 添加边到邻接表
    public void addEdge(int source, int destination) {
        adjacencyList[source].add(destination);
    }

    // 广度优先搜索遍历
    public void bfsTraversal(int startNode) {
        Queue<Integer> queue = new LinkedList<>(); // 创建一个队列用于BFS遍历
        queue.add(startNode); // 将起始节点加入队列
        visited[startNode] = true; // 标记起始节点为已访问

        while (!queue.isEmpty()) {
            int currentNode = queue.poll(); // 出队列一个节点
            System.out.print(currentNode + " "); // 输出当前节点

            for (int neighbor : adjacencyList[currentNode]) {
                if (!visited[neighbor]) {
                    queue.add(neighbor); // 将未访问的邻居节点加入队列
                    visited[neighbor] = true; // 标记邻居节点为已访问
                }
            }
        }
    }
}
相关推荐
初晴や16 小时前
【C++】图论:基础理论与实际应用深入解析
c++·算法·图论
闻缺陷则喜何志丹21 小时前
【图论 DFS 换根法】3772. 子图的最大得分|2235
c++·算法·深度优先·力扣·图论·换根法
君义_noip2 天前
信息学奥赛一本通 2134:【25CSPS提高组】道路修复 | 洛谷 P14362 [CSP-S 2025] 道路修复
c++·算法·图论·信息学奥赛·csp-s
罗湖老棍子3 天前
信使(msner)(信息学奥赛一本通- P1376)四种做法
算法·图论·dijkstra·spfa·floyd·最短路算法
修炼地3 天前
代码随想录算法训练营第五十三天 | 卡码网97. 小明逛公园(Floyd 算法)、卡码网127. 骑士的攻击(A * 算法)、最短路算法总结、图论总结
c++·算法·图论
罗湖老棍子3 天前
【例4-6】香甜的黄油(信息学奥赛一本通- P1345)
算法·图论·dijkstra·floyd·最短路算法·bellman ford
--JR5 天前
015——图(1.图的相关概念与存储)
数据结构·c++·算法·链表·图论
闻缺陷则喜何志丹5 天前
【二分查找 图论】P10206 [JOI 2024 Final] 建设工程 2|普及+
c++·算法·二分查找·图论·洛谷
surtr17 天前
全源最短路封装模板(APSP,Floyd求最小环,Floyd求最短路,Johnson算法)
c++·算法·数学建模·动态规划·图论
surtr18 天前
【算法自用】一些比较有趣的题目
算法·动态规划·概率论·图论