-
Notifications
You must be signed in to change notification settings - Fork 106
/
Copy pathfind-if-path-exists-in-graph.cpp
135 lines (129 loc) · 3.75 KB
/
find-if-path-exists-in-graph.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
// Time: O(|V| + |E|)
// Space: O(|V| + |E|)
// bi-bfs solution
class Solution {
public:
bool validPath(int n, vector<vector<int>>& edges, int start, int end) {
unordered_map<int, vector<int>> adj;
for (const auto& edge : edges) {
adj[edge[0]].emplace_back(edge[1]);
adj[edge[1]].emplace_back(edge[0]);
}
return bi_bfs(adj, start, end) >= 0;
}
private:
int bi_bfs(const unordered_map<int, vector<int>>& adj,
const int start,
const int target) {
unordered_set<int> left = {start}, right = {target};
unordered_set<int> lookup;
int steps = 0;
while (!empty(left)) {
for (const auto& pos : left) {
lookup.emplace(pos);
}
unordered_set<int> new_left;
for (const auto& pos : left) {
if (right.count(pos)) {
return steps;
}
if (!adj.count(pos)) {
continue;
}
for (const auto& nei : adj.at(pos)) {
if (lookup.count(nei)) {
continue;
}
new_left.emplace(nei);
}
}
left = move(new_left);
++steps;
if (size(left) > size(right)) {
swap(left, right);
}
}
return -1;
}
};
// Time: O(|V| + |E|)
// Space: O(|V| + |E|)
// bfs solution
class Solution2 {
public:
bool validPath(int n, vector<vector<int>>& edges, int start, int end) {
unordered_map<int, vector<int>> adj;
for (const auto& edge : edges) {
adj[edge[0]].emplace_back(edge[1]);
adj[edge[1]].emplace_back(edge[0]);
}
return bfs(adj, start, end) >= 0;
}
private:
int bfs(const unordered_map<int, vector<int>>& adj,
const int start,
const int target) {
vector<int> q = {start};
unordered_set<int> lookup;
int steps = 0;
while (!empty(q)) {
vector<int> new_q;
for (const auto& pos : q) {
if (pos == target) {
return steps;
}
if (!adj.count(pos)) {
continue;
}
for (const auto& nei : adj.at(pos)) {
if (lookup.count(nei)) {
continue;
}
lookup.emplace(nei);
new_q.emplace_back(nei);
}
}
q = move(new_q);
++steps;
}
return -1;
}
};
// Time: O(|V| + |E|)
// Space: O(|V| + |E|)
// dfs solution
class Solution3 {
public:
bool validPath(int n, vector<vector<int>>& edges, int start, int end) {
unordered_map<int, vector<int>> adj;
for (const auto& edge : edges) {
adj[edge[0]].emplace_back(edge[1]);
adj[edge[1]].emplace_back(edge[0]);
}
return dfs(adj, start, end);
}
private:
int dfs(const unordered_map<int, vector<int>>& adj,
const int start,
const int target) {
vector<int> stk = {start};
unordered_set<int> lookup;
while (!empty(stk)) {
auto pos = stk.back(); stk.pop_back();
if (pos == target) {
return true;
}
if (!adj.count(pos)) {
continue;
}
for (const auto& nei : adj.at(pos)) {
if (lookup.count(nei)) {
continue;
}
lookup.emplace(nei);
stk.emplace_back(nei);
}
}
return false;
}
};