描述
有一个 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
}
}