day48 代码随想录算法训练营 图论专题1

1 今日打卡

深度优先搜索

可达路径 98. 可达路径

广度优先搜索

2 深度优先搜索

代码框架:

3 可达路径

3.1 思路

  1. 数据结构与初始化

邻接矩阵(int [][] graph):用二维数组存储图的边关系,graph[from][to] = 1 表示从from节点到to节点有一条有向边。

全局集合(res):存储所有找到的有效路径(从 1 到 n 的完整路径)。

全局路径(path):记录 DFS 过程中当前正在遍历的路径,通过 "添加 - 递归 - 移除" 的回溯逻辑复用该集合。

  1. DFS 核心逻辑(深度优先搜索 + 回溯)

终止条件:当当前遍历的节点cur等于目标节点des(即节点 n)时,说明找到了一条完整路径,将当前path的副本存入结果集res(必须用new ArrayList<>(path),否则存的是引用,后续回溯会修改结果)。

遍历与递归:对当前节点cur的所有邻接节点(通过邻接矩阵graph[cur][i] == 1判断),执行 3 个操作:

加入路径:将邻接节点i加入path;

递归深入:以i为新的当前节点,继续 DFS;

回溯移除:递归返回后,移除path最后一个节点(恢复现场),尝试其他邻接节点。

  1. 主函数流程

读取输入(节点数 n、边数 m),构建邻接矩阵;

初始化起始路径(将节点 1 加入path,因为路径从 1 开始);

调用 DFS 遍历所有路径;

输出结果:无路径则输出 - 1,有路径则按 "节点 节点 ... 节点" 格式打印每条路径。

3.2 实现代码

java 复制代码
import java.util.*;

public class Main {
    // 存储所有从节点1到节点n的有效路径
    static List<List<Integer>> res = new ArrayList<>();
    // 存储DFS过程中当前遍历的路径(回溯复用)
    static List<Integer> path = new ArrayList<>();

    /**
     * 深度优先搜索(DFS)找所有路径
     * @param graph 邻接矩阵,graph[cur][i]=1表示cur到i有边
     * @param cur 当前遍历的节点
     * @param des 目标节点(即节点n)
     */
    public static void dfs(int[][] graph, int cur, int des) {
        // 终止条件:当前节点到达目标节点,找到一条完整路径
        if (cur == des) {
            // 存入path的副本(避免后续回溯修改结果)
            res.add(new ArrayList<>(path));
            return;
        }

        // 遍历当前节点的所有可能邻接节点(节点编号从1到n)
        for (int i = 1; i < graph.length; i++) {
            // 判断cur到i是否有有向边
            if (graph[cur][i] == 1) {
                path.add(i);          // 将i加入当前路径
                dfs(graph, i, des);   // 递归遍历i的邻接节点
                path.remove(path.size() - 1); // 回溯:移除最后一个节点,尝试其他分支
            }
        }
    }

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        // 读取节点数n、边数m(节点编号:1~n)
        int n = sc.nextInt(), m = sc.nextInt();
        // 初始化邻接矩阵:n+1行n+1列(避免下标越界,不用0号位置)
        int[][] graph = new int[n+1][n+1];

        // 构建邻接矩阵
        for (int i = 0; i < m; i++) {
            int from = sc.nextInt(); // 边的起点
            int to = sc.nextInt();   // 边的终点
            graph[from][to] = 1;     // 标记有向边
        }

        // 路径从节点1开始,先将1加入path
        path.add(1);
        // 调用DFS:从节点1出发,找到达节点n的所有路径
        dfs(graph, 1, n);

        // 输出结果
        if (res.size() == 0) {
            // 无有效路径时输出-1
            System.out.println(-1);
        } else {
            // 遍历所有有效路径并打印
            for (List<Integer> singlePath : res) {
                // 打印路径的前n-1个节点(带空格)
                for (int i = 0; i < singlePath.size() - 1; i++) {
                    System.out.print(singlePath.get(i) + " ");
                }
                // 打印最后一个节点(换行,避免末尾多余空格)
                System.out.println(singlePath.get(singlePath.size() - 1));
            }
        }
        sc.close(); // 关闭扫描器,释放资源
    }
}

4 广度优先搜索

从start起点开始,是一圈一圈,向外搜索

正是因为BFS一圈一圈的遍历方式,所以一旦遇到终止点,那么一定是一条最短路径。

一圈的待遍历节点,放到容器里保存起来。栈、队列甚至数组都可以。

相关推荐
架构师沉默2 小时前
如果 Spring 没了,Java 会怎么样?
java·后端·架构
absunique2 小时前
多路归并算法在外部排序中的实现与优化的技术7
算法
叶宇燚2 小时前
Java整理--数据结构篇
java·开发语言·数据结构
鹿鸣悠悠2 小时前
【AI-08】Prompt(提示词)
人工智能·算法
数据中穿行2 小时前
12种经典排序算法完整C++实现
算法
2501_926978332 小时前
从“数字果蝇” 到“数字意识”以及AGI的构建思路
大数据·人工智能·经验分享·算法·ai写作·agi
吴声子夜歌2 小时前
小程序——界面API(一)
java·javascript·小程序
予枫的编程笔记2 小时前
【面试专栏|Java并发编程】从Runnable到Callable,Java4种线程创建方式
java·多线程·thread·java面试·runnable·callable·java线程
无尽的罚坐人生2 小时前
hot 100 101. 对称二叉树
数据结构·算法·leetcode