0

0

BFS遍历矩阵迷宫时陷入无限循环的解决方案

霞舞

霞舞

发布时间:2025-12-26 17:41:02

|

957人浏览过

|

来源于php中文网

原创

BFS遍历矩阵迷宫时陷入无限循环的解决方案

bfs在矩阵迷宫中搜索路径时,若未标记已访问节点,会导致同一位置反复入队,引发队列持续增长、永不为空的死循环。关键在于引入访问标记机制,避免重复访问。

在您提供的代码中,BFS逻辑结构基本正确:从起点入队,逐层扩展上下左右四个方向的合法邻居,并通过 Box 节点的 parent 字段回溯路径。但核心缺陷在于缺少访问状态管理——程序未记录哪些坐标已被探索过,导致节点反复被重新加入队列。

例如,从 (0,0) 扩展到 (0,1) 后,当 (0,1) 出队时又会检查 (0,0)(左邻),而此时 (0,0) 仍满足 isFree() 条件(值为 0 且坐标合法),于是重新入队。如此往复,形成环路,队列无限膨胀。

✅ 正确做法是:每弹出一个节点(poll())后,立即标记其坐标为“已访问”;后续所有邻居入队前,必须同时校验边界、可通行性 未访问状态。

推荐使用二维布尔数组 visited[][] 实现标记(也可复用原矩阵做原地标记,但需谨慎处理):

ChatWP
ChatWP

一个AI聊天机器人,可以直接回答你的WordPress问题。

下载
public static void searchPath(int[][] maze, int startX, int startY, ArrayList path) {
    int rows = maze.length;
    int cols = maze[0].length;
    boolean[][] visited = new boolean[rows][cols]; // 新增访问标记数组
    Queue queue = new LinkedList<>(); // 避免静态变量,防止多线程/多次调用冲突
    queue.add(new Box(startX, startY, null));
    visited[startY][startX] = true; // 入队即标记

    while (!queue.isEmpty()) {
        Box p = queue.poll();

        if (maze[p.y][p.x] == 9) {
            System.out.println("Exit is reached!");
            getPath(p, maze, path);
            return;
        }

        // 四方向扩展(顺序无关紧要)
        int[][] directions = {{1,0}, {-1,0}, {0,1}, {0,-1}};
        for (int[] dir : directions) {
            int nx = p.x + dir[0];
            int ny = p.y + dir[1];
            if (isFree(maze, nx, ny) && !visited[ny][nx]) {
                visited[ny][nx] = true; // 入队前标记,防重复
                queue.add(new Box(nx, ny, p));
            }
        }
    }
    System.out.println("No path found.");
}

// isFree 保持不变,但注意:原实现中 maze[y][x] 的索引顺序需与实际矩阵维度一致(此处假设 maze[y][x] 正确)
public static boolean isFree(int[][] maze, int x, int y) {
    if (x >= 0 && x < maze[0].length && y >= 0 && y < maze.length) {
        int val = maze[y][x];
        return val == 0 || val == 2 || val == 9; // 仅允许通路、路径标记、终点
    }
    return false;
}

⚠️ 重要注意事项

  • 避免静态队列:原代码中 public static Queue q 是危险设计。静态变量跨调用持久化,易引发状态污染;应改为局部变量或通过参数传递。
  • 索引一致性:maze[y][x] 表示第 y 行、第 x 列,需确保 maze 是 rows × cols 结构(即 maze.length 为行数,maze[0].length 为列数)。isFree() 中 maze[y][x] 的行列顺序必须与矩阵实际布局严格匹配。
  • 标记时机:务必在 queue.add() 之前 设置 visited[ny][nx] = true,否则在高并发或多分支场景下仍可能因竞态导致重复入队。
  • 内存与性能:对大型迷宫,visited 数组是必要开销;若内存受限,可考虑将已访问位置在 maze 中临时设为 -1 等非法值(但需保证不影响 isFree() 判断逻辑)。

总结:BFS 的本质是层级遍历,其正确性依赖于每个节点仅被处理一次。缺失访问标记是初学者最常犯的错误之一。加上 visited 数组并规范标记时机,即可彻底解决死循环问题,确保算法在 O(M×N) 时间内稳定完成搜索。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

899

2023.09.19

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

381

2023.08.14

虚拟号码教程汇总
虚拟号码教程汇总

本专题整合了虚拟号码接收验证码相关教程,阅读下面的文章了解更多详细操作。

30

2025.12.25

错误代码dns_probe_possible
错误代码dns_probe_possible

本专题整合了电脑无法打开网页显示错误代码dns_probe_possible解决方法,阅读专题下面的文章了解更多处理方案。

20

2025.12.25

网页undefined啥意思
网页undefined啥意思

本专题整合了undefined相关内容,阅读下面的文章了解更多详细内容。后续继续更新。

37

2025.12.25

word转换成ppt教程大全
word转换成ppt教程大全

本专题整合了word转换成ppt教程,阅读专题下面的文章了解更多详细操作。

6

2025.12.25

msvcp140.dll丢失相关教程
msvcp140.dll丢失相关教程

本专题整合了msvcp140.dll丢失相关解决方法,阅读专题下面的文章了解更多详细操作。

2

2025.12.25

笔记本电脑卡反应很慢处理方法汇总
笔记本电脑卡反应很慢处理方法汇总

本专题整合了笔记本电脑卡反应慢解决方法,阅读专题下面的文章了解更多详细内容。

6

2025.12.25

微信调黑色模式教程
微信调黑色模式教程

本专题整合了微信调黑色模式教程,阅读下面的文章了解更多详细内容。

5

2025.12.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号