01矩阵

问题描述

给定矩阵,求出每个元素离最近的 0 的距离,注意最近的 0 可以不和该元素同一行

输入案例

0 0 0
0 1 0
1 1 1

输出

0 0 0
0 1 0
1 2 1

解决思路

  咋一看貌似可以用DP来做哎,如果知道了一个元素的附近邻居的最小距离,然后取邻居的最小距离+1不就出来了嘛。大体思路确实如此,但是怎样获得邻居元素的结果呢?

具体思路

  我们在遍历的时候总是根据从左到右、从上到下的顺序来的,对于某个我们想要考虑的元素,如果他的左边邻居和上面的邻居都已经被计算过,那么我们就可以根据他的左、上边邻居得到他的计算结果值。第一次遍历结束后,还只是考虑了从左上邻居的情况,没有考虑右下邻居的情况,然后再进行一次遍历,从右往左、从下往上遍历,此时就可以根据右、下邻居计算出一个值,这个值要和第一遍根据左上邻居计算出来的结果进行比较取小。即得到该位置处的计算结果。

  可能有同学会问了,为什么不能用一次遍历呢,直接把该元素的上下左右四个邻居全部考虑进来呢?对于这个问题,我肯定是摔过跟头的啊!我们可以考虑这样一点,从上到下,从左到右遍历的时候,对于给定位置的元素,他的左上邻居才有计算结果,右下邻居根本还没得到计算,怎么可以把非计算结果代带进去比较呢?举个例子,比如矩阵中的某个小区域是这个样子:

1 0 1 1
1 1 1 1
0 0 1 1
1 0 1 1

只遍历一次的前两行结果为:
1 0 1 2
1 1 2 2

  对于第2行最后一个结果,明显错了对不对,因为他直接把他的下邻居的 1 拿来计算了。

Java 实现

 1 class Solution {
 2     private static int[] dx = {-1, 0, 1, 0}; //  上 左 下 右
 3     private static int[] dy = {0, -1, 0, 1};
 4     
 5     public static int[][] updateMatrix(int[][] matrix) {
 6         int m = matrix.length, n = matrix[0].length;
 7         for (int i = 0; i < m; i++) {
 8             for (int j = 0; j < n; j++) {
 9                 if (matrix[i][j] == 0) continue;
10                 int newI, newJ, min=10001;
11                 // 左、上邻居
12                 for (int k = 0; k < 2; k++) {
13                     newI = i + dx[k];
14                     newJ = j + dy[k];
15                     if (0 <= newI && newI < m && 0 <= newJ && newJ < n) {
16                         min = Math.min(min, matrix[newI][newJ]);
17                     }
18                 }
19                 matrix[i][j] = min + 1;
20             }
21         }
22         for (int i = m-1; i >=0; i--) {
23             for (int j = n-1; j >=0; j--) {
24                 if (matrix[i][j] == 0) continue;
25                 int newI, newJ, min=10001;
26                 // 右 下邻居
27                 for (int k = 2; k < 4; k++) {
28                     newI = i + dx[k];
29                     newJ = j + dy[k];
30                     if (0 <= newI && newI < m && 0 <= newJ && newJ < n) {
31                         min = Math.min(min, matrix[newI][newJ]);
32                     }
33                 }
34                 matrix[i][j] = Math.min(matrix[i][j], min + 1);
35             }
36         }
37         return matrix;
38     }
39 }        

 

posted @ 2019-08-19 17:32  LimLee  阅读(536)  评论(0编辑  收藏  举报