-
Notifications
You must be signed in to change notification settings - Fork 521
/
Copy pathShortest Distance from All Buildings.cpp
137 lines (112 loc) · 4.35 KB
/
Shortest Distance from All Buildings.cpp
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
/*
Shortest Distance from All Buildings
====================================
You are given an m x n grid grid of values 0, 1, or 2, where:
each 0 marks an empty land that you can pass by freely,
each 1 marks a building that you cannot pass through, and
each 2 marks an obstacle that you cannot pass through.
You want to build a house on an empty land that reaches all buildings in the shortest total travel distance. You can only move up, down, left, and right.
Return the shortest travel distance for such a house. If it is not possible to build such a house according to the above rules, return -1.
The total travel distance is the sum of the distances between the houses of the friends and the meeting point.
The distance is calculated using Manhattan Distance, where distance(p1, p2) = |p2.x - p1.x| + |p2.y - p1.y|.
Example 1:
Input: grid = [[1,0,2,0,1],[0,0,0,0,0],[0,0,1,0,0]]
Output: 7
Explanation: Given three buildings at (0,0), (0,4), (2,2), and an obstacle at (0,2).
The point (1,2) is an ideal empty land to build a house, as the total travel distance of 3+3+1=7 is minimal.
So return 7.
Example 2:
Input: grid = [[1,0]]
Output: 1
Example 3:
Input: grid = [[1]]
Output: -1
Constraints:
m == grid.length
n == grid[i].length
1 <= m, n <= 50
grid[i][j] is either 0, 1, or 2.
There will be at least one building in the grid.
*/
class Solution {
private:
int bfs(vector<vector<int>>& grid, int row, int col, int totalHouses) {
// Next four directions.
int dirs[4][2] = {{1, 0}, {-1, 0}, {0, 1}, {0, -1}};
int rows = grid.size();
int cols = grid[0].size();
int distanceSum = 0;
int housesReached = 0;
// Queue to do a bfs, starting from (r,c) cell
queue<pair<int, int>> q;
q.push({ row, col });
// Keep track of visited cells
vector<vector<bool>> vis(rows, vector<bool> (cols, false));
vis[row][col] = true;
int steps = 0;
while (!q.empty() && housesReached != totalHouses) {
for (int i = q.size(); i > 0; --i) {
auto curr = q.front();
q.pop();
row = curr.first;
col = curr.second;
// If this cell is a house, then add the distance from the source to this cell
// and we go past from this cell.
if (grid[row][col] == 1) {
distanceSum += steps;
housesReached++;
continue;
}
// This cell was an empty cell, hence traverse the next cells which is not a blockage.
for (auto& dir : dirs) {
int nextRow = row + dir[0];
int nextCol = col + dir[1];
if (nextRow >= 0 && nextCol >= 0 && nextRow < rows && nextCol < cols) {
if (!vis[nextRow][nextCol] && grid[nextRow][nextCol] != 2) {
vis[nextRow][nextCol] = true;
q.push({nextRow, nextCol});
}
}
}
}
// After traversing one level cells, increment the steps by 1 to reach to next level.
steps++;
}
if (housesReached != totalHouses) {
for (row = 0; row < rows; row++) {
for (col = 0; col < cols; col++) {
if (grid[row][col] == 0 && vis[row][col]) {
grid[row][col] = 2;
}
}
}
return INT_MAX;
}
return distanceSum;
}
public:
int shortestDistance(vector<vector<int>>& grid) {
int minDistance = INT_MAX;
int rows = grid.size();
int cols = grid[0].size();
int totalHouses = 0;
for (int row = 0; row < rows; ++row) {
for (int col = 0; col < cols; ++col) {
if (grid[row][col] == 1) {
totalHouses++;
}
}
}
for (int row = 0; row < rows; ++row) {
for (int col = 0; col < cols; ++col) {
if (grid[row][col] == 0) {
minDistance = min(minDistance, bfs(grid, row, col, totalHouses));
}
}
}
if (minDistance == INT_MAX) {
return -1;
}
return minDistance;
}
};