HJ43 迷宫问题

描述

迷宫问题_牛客题霸_牛客网

有一个 hh 行 ww 列的网格,我们使用 (i,j)(i,j) 表示网格中从上往下数第 ii 行和从左往右数第 jj 列的单元格。每个方格要么是可以通过的空方格 '0''0' ,要么是不可通过的墙方格 '1''1' ,特别的,网格的四周都是墙方格,你可以沿着空方格上下左右随意移动:从 (x,y)(x,y) 向上移动一格即抵达 (x−1,y)(x−1,y) 、向下移动一格即抵达 (x+1,y)(x+1,y) 、向左移动一格即抵达 (x,y−1)(x,y−1) 、向右移动一格即抵达 (x,y+1)(x,y+1) 。

现在,你位于迷宫的入口 (0,0)(0,0) ,想要前往终点 (h−1,w−1)(h−1,w−1) 。请输出一条从起点到终点的可行路径。

保证起点和终点一定为空方格,你始终可以找到且能唯一找到一条从起点出发到达终点的可行路径。

输入描述:

第一行输入两个整数 h,w(1≦h,w≦100)h,w(1≦h,w≦100) 代表迷宫的行数和列数。

此后 hh 行,第 ii 行输入 ww 个整数 ai,1,ai,2,...,ai,w(0≦ai,j≦1)ai,1​,ai,2​,...,ai,w​(0≦ai,j​≦1) 代表迷宫的布局。其中,ai,j=0ai,j​=0 表示单元格 (i,j)(i,j) 是空方格,ai,j=1ai,j​=1 表示单元格 (i,j)(i,j) 是墙方格。

输出描述:

输出若干行,第 ii 行输出两个整数 xi,yixi​,yi​ ,表示路径的第 ii 步抵达的单元格坐标为 (xi,yi)(xi​,yi​) 。

你需要保证输出的路径是符合题目要求的,即从起点 (0,0)(0,0) 出发,到达终点 (h−1,w−1)(h−1,w−1) ,且路径上每个单元格都是空方格,行走的单元格都是彼此相邻的。

输入:

复制代码
5 5
0 1 0 0 0
0 1 1 1 0
0 0 0 0 0
0 1 1 1 0
0 0 0 1 0

输出:

复制代码
(0,0)
(1,0)
(2,0)
(2,1)
(2,2)
(2,3)
(2,4)
(3,4)
(4,4)

输入:

复制代码
5 5
0 1 0 0 0
0 1 0 1 0
0 0 0 0 1
0 1 1 1 0
0 0 0 0 0

输出:

复制代码
(0,0)
(1,0)
(2,0)
(3,0)
(4,0)
(4,1)
(4,2)
(4,3)
(4,4)

代码思路

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

public class Main {

    public static void main(String[] args) {
        Scanner in = new Scanner(System.in);
        int m = in.nextInt(); // 地图的行数
        int n = in.nextInt(); // 地图的列数
        int[][] nums = new int[m][n]; // 存储地图信息,0表示通路,1表示障碍物
        for (int i = 0; i < m; i++) {
            for (int j = 0; j < n; j++) {
                nums[i][j] = in.nextInt(); // 读入地图内容
            }
        }

        // 用于记录当前路径的坐标序列
        List<Integer[]> list = new ArrayList<>();
        list.add(new Integer[]{0, 0}); // 起点加入路径

        boolean[][] visit = new boolean[m][n]; // 标记某个位置是否已经访问过,避免重复搜索

        // 调用 DFS 寻找路径
        dfs(list, nums, visit, m, n, 0, 0);

        // 移除起点最后一次重复记录
        list.remove(list.size() - 1);

        // 输出路径中所有坐标
        for (Integer[] path : list) {
            System.out.println("(" + path[0] + "," + path[1] + ")");
        }
    }

    // 定义四个方向:下、上、右、左
    public static Integer[][] direction = {{1, 0}, {-1, 0}, {0, 1}, {0, -1}};

    /**
     * DFS 方法,递归搜索路径
     * @param list 当前路径上的坐标列表
     * @param nums 地图矩阵
     * @param visit 标记数组,标记访问过的位置
     * @param m 地图行数
     * @param n 地图列数
     * @param x 当前行坐标
     * @param y 当前列坐标
     * @return 是否找到从起点到终点的路径
     */
    public static boolean dfs(List<Integer[]> list, int[][] nums, boolean[][] visit, int m, int n, int x, int y) {
        // 边界条件和障碍判断
        if (x < 0 || x >= m || y < 0 || y >= n || nums[x][y] == 1 || visit[x][y]) {
            // 这里判断是否到达终点(右下角)
            if (x == m && y == n - 1 || x == m - 1 && y == n) {
                return true;
            } else {
                return false; // 越界或障碍,返回false
            }
        }

        visit[x][y] = true; // 标记当前位置已访问

        // 遍历四个方向
        for (Integer[] dir : direction) {
            int newx = x + dir[0];
            int newy = y + dir[1];
            list.add(new Integer[]{newx, newy}); // 将新坐标加入路径
            if (dfs(list, nums, visit, m, n, newx, newy)) {
                return true; // 找到一条通路,立即返回
            }
            list.remove(list.size() - 1); // 回溯,移除当前方向坐标
        }

        return false; // 四个方向都没找到通路,返回false
    }
}
相关推荐
小龙报1 小时前
【算法通关指南:数据结构与算法篇(五)】树的 “自我介绍”:从递归定义到存储绝技(vector vs 链式前向星)
c语言·数据结构·c++·算法·链表·启发式算法·visual studio
报错小能手1 小时前
数据结构 顺序栈
数据结构·算法
点云SLAM1 小时前
C++包装器之类型擦除(Type Erasure)包装器详解(4)
c++·算法·c++17·类型擦除·c++高级应用·c++包装器·函数包装
yugi9878381 小时前
TDOA算法MATLAB实现:到达时间差定位
前端·算法·matlab
t198751281 小时前
基于因子图与和积算法的MATLAB实现
开发语言·算法·matlab
le serein —f1 小时前
用go实现-回文链表
算法·leetcode·golang
rit84324992 小时前
MFOCUSS算法MATLAB实现:稀疏信号重构
算法·matlab·重构
发疯幼稚鬼2 小时前
散列及其分离链接法
c语言·数据结构·算法·链表·散列表
Bdygsl2 小时前
数字图像处理总结 Day 1
人工智能·算法·计算机视觉