题目
你现在手里有一份大小为 N x N 的『地图』(网格) grid,上面的每个『区域』(单元格)都用 0 和 1 标记好了。其中 0 代表海洋,1 代表陆地,你知道距离陆地区域最远的海洋区域是是哪一个吗?请返回该海洋区域到离它最近的陆地区域的距离。
我们这里说的距离是『曼哈顿距离』( Manhattan Distance):(x0, y0) 和 (x1, y1) 这两个区域之间的距离是 |x0 - x1| + |y0 - y1| 。
如果我们的地图上只有陆地或者海洋,请返回 -1。
链接:https://leetcode-cn.com/problems/as-far-from-land-as-possible
示例 1:
输入:[[1,0,1],[0,0,0],[1,0,1]]
输出: 2
解释:
海洋区域 (1, 1) 和所有陆地区域之间的距离都达到最大,最大距离为 2。
示例 2:
输入:[[1,0,0],[0,0,0],[0,0,0]]
输出: 4
解释:
海洋区域 (2, 2) 和所有陆地区域之间的距离都达到最大,最大距离为 4。
提示:
- 1 <= grid.length == grid[0].length <= 100
- grid[i][j] 不是 0 就是 1
解答
本题思路类似以前做过的迷宫问题,是BFS。
class Solution {
//横向移动和纵向移动1格所需的坐标
static int[] x = new int[]{1,-1,0,0};
static int[] y = new int[]{0,0,1,-1};
public static int maxDistance(int[][] grid) {
//初始化队列、最大距离
int distance = 0;
int n = grid.length;
Queue<int[]> queue = new LinkedList<>();
//初始化记录
for (int i = 0;i < n;i++){
for (int j = 0;j < n;j++){
if (grid[i][j] == 1){
//放入队列的横纵坐标和开始距离
queue.offer(new int[]{i,j,0});
}
}
}
//全是0或全是1
if (queue.isEmpty() || queue.size() == n*n){
return -1;
}
while(!queue.isEmpty()){
//提取队列中的陆地
int[] temp = queue.poll();
int x1 = temp[0];
int y1 = temp[1];
int distance1 = temp[2];
distance = Math.max(distance,distance1);
//四方位移动
for (int i = 0;i < 4;i++){
int x2 = x1 + x[i];
int y2 = y1 + y[i];
if (x2>=0 && x2<n && y2>=0 && y2<n && grid[x2][y2]==0){
//改海洋为陆地
grid[x2][y2] = 1;
queue.offer(new int[]{x2,y2,distance1+1});
}
}
}
return distance;
}
}