-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathWalls and Gates.java
executable file
·191 lines (153 loc) · 6.41 KB
/
Walls and Gates.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
M
1533438636
tags: BFS, DFS
给一个room 2D grid. 里面有墙-1, 门0, 还有empty space INF(Math.MAX_VALUE).
对每个empty space而言, fill it with dist to nearest gate.
#### DFS
- Form empty room: it can reach different gate, but each shortest length will be determined by the 4 directions.
- Option1(NOT applicable). DFS on INF, mark visited, summerize results of 4 directions.
- hard to resue: we do not know the direction in cached result dist[i][j]
- Option2. DFS on gate, and each step taken to each direction will +1 on the spot: distance from one '0';
- Through dfs from all zeros, update each spot with shorter dist
- Worst time: O(mn), where entre rooms[][] are gates. It takes O(mn) to complete the iteration. Other gates be skipped by `if (rooms[x][y] <= dist) return;`
#### BFS
- Exact same concept. Init with `Queue<int[]> queue = new LinkedList<int[]>()`
```
/*
You are given a m x n 2D grid initialized with these three possible values.
-1 - A wall or an obstacle.
0 - A gate.
INF - Infinity means an empty room. We use the value 231 - 1 = 2147483647
to represent INF as you may assume that the distance to a gate is less than 2147483647.
Fill each empty room with the distance to its nearest gate.
If it is impossible to reach a gate, it should be filled with INF.
Example:
Given the 2D grid:
INF -1 0 INF
INF INF INF -1
INF -1 INF -1
0 -1 INF INF
After running your function, the 2D grid should be:
3 -1 0 1
2 2 1 -1
1 -1 2 -1
0 -1 3 4
*/
// No need of the visited. Simplified code:
class Solution {
int[] dx = {1, -1, 0, 0};
int[] dy = {0, 0, 1, -1};
public void wallsAndGates(int[][] rooms) {
if (validate(rooms)) return;
int m = rooms.length, n = rooms[0].length;
boolean[][] visited = new boolean[m][n];
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (rooms[i][j] == 0) {// test all 0's with for loop
dfsHelper(rooms, i, j, 1);
}
}
}
}
// mark grid with dist (compare if existed)
public void dfs(int[][] rooms, int x, int y, int dist) {
if (validateCoordinate(rooms, x, y)) return;
if (rooms[x][y] <= dist) return;
rooms[x][y] = dist; // dist < room[x][y], so update rooms[x][y] with smaller dist
dfsHelper(rooms, x, y, dist + 1); // dfs if applicable
}
private void dfsHelper(int[][] rooms, int x, int y, int dist) {
for (int i = 0; i < dx.length; i++) {
dfs(rooms, x + dx[i], y + dy[i], dist);
}
}
private boolean validateCoordinate(int[][] rooms, int x, int y) {
return x < 0 || x >= rooms.length || y < 0 || y >= rooms[0].length || rooms[x][y] == -1 || rooms[x][y] == 0;
}
private boolean validate(int[][] rooms) {
return rooms == null || rooms.length == 0 || rooms[0] == null || rooms[0].length == 0;
}
}
// BFS:
class Solution {
int[] dx = {1, -1, 0, 0};
int[] dy = {0, 0, 1, -1};
public void wallsAndGates(int[][] rooms) {
if (validate(rooms)) return;
int m = rooms.length, n = rooms[0].length;
Queue<int[]> queue = new LinkedList<>();
// Initi with 0
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (rooms[i][j] == 0) {// test all 0's with for loop
queue.offer(new int[] {i, j});
}
}
}
// Process queue
while (!queue.isEmpty()) {
int[] point = queue.poll();
bfsHelper(rooms, queue, point[0], point[1]);
}
}
// Process with queue, and skip of value depth > existing depth.
private void bfsHelper(int[][] rooms, Queue<int[]> queue, int x, int y) {
for (int i = 0; i < dx.length; i++) {
int mX = x + dx[i], mY = y + dy[i];
if (validateCoordinate(rooms, mX, mY) || rooms[x][y] + 1 > rooms[mX][mY]) continue;
rooms[mX][mY] = rooms[x][y] + 1;
queue.offer(new int[] {mX, mY});
}
}
private boolean validateCoordinate(int[][] rooms, int x, int y) {
return x < 0 || x >= rooms.length || y < 0 || y >= rooms[0].length || rooms[x][y] == -1 || rooms[x][y] == 0;
}
private boolean validate(int[][] rooms) {
return rooms == null || rooms.length == 0 || rooms[0] == null || rooms[0].length == 0;
}
}
/*
Form empty room: it can reach different gate, but each shortest length will be determined by the 4 directions.
Option1. DFS on INF, mark visited, summerize results of 4 directions. it's hard to resue: we do not know the direction in cached result dist[i][j]
Option2. DFS on gate, and each step taken to each direction will +1 on the spot: distance from one '0'; through dfs from all zeros, update each spot with shorter dist
*/
class Solution {
public void wallsAndGates(int[][] rooms) {
// check input
if (validate(rooms)) return;
// find all 0's with for loop
int m = rooms.length, n = rooms[0].length;
boolean[][] visited = new boolean[m][n];
for (int i = 0; i < m; i++) {
for (int j = 0; j < n; j++) {
if (rooms[i][j] == 0 && !visited[i][j]) {
runDfs(rooms, visited, i, j, 0);
}
}
}
}
// mark visited with dist (compare if existed)
public void dfs(int[][] rooms, boolean[][] visited, int x, int y, int dist) {
if (validateCoordinate(rooms, x, y)) return;
if (rooms[x][y] <= dist) return;
// update rooms[x][y]
visited[x][y] = true;
rooms[x][y] = dist; //Math.min(rooms[x][y], dist);
// dfs if applicable
runDfs(rooms, visited, x, y, dist);
visited[x][y] = false;
}
private void runDfs(int[][] rooms, boolean[][] visited, int x, int y, int dist) {
dfs(rooms, visited, x-1, y, dist + 1);
dfs(rooms, visited, x+1, y, dist + 1);
dfs(rooms, visited, x, y-1, dist + 1);
dfs(rooms, visited, x, y+1, dist + 1);
}
private boolean validateCoordinate(int[][] rooms, int x, int y) {
return x < 0 || x >= rooms.length || y < 0 || y >= rooms[0].length || rooms[x][y] == -1 || rooms[x][y] == 0;
}
private boolean validate(int[][] rooms) {
return rooms == null || rooms.length == 0 || rooms[0] == null || rooms[0].length == 0;
}
}
```