-
Notifications
You must be signed in to change notification settings - Fork 1.6k
/
Copy pathtree-of-coprimes.cpp
103 lines (99 loc) · 3.48 KB
/
tree-of-coprimes.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
// Time: O(50 * n) = O(n)
// Space: O(n)
class Solution {
public:
vector<int> getCoprimes(vector<int>& nums, vector<vector<int>>& edges) {
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 iter_dfs(nums, adj);
}
private:
vector<int> iter_dfs(const vector<int>& nums, const unordered_map<int, vector<int>>& adj) {
vector<int> result(size(nums), -1);
unordered_map<int, vector<pair<int, int>>> path;
vector<tuple<int, int, int, int>> stk = {{1, -1, 0, 0}};
while (!empty(stk)) {
const auto [step, prev, node, depth] = stk.back(); stk.pop_back();
if (step == 1) {
stk.emplace_back(4, -1, node, -1);
stk.emplace_back(3, prev, node, depth);
stk.emplace_back(2, -1, node, -1);
} else if (step == 2) {
int max_d = -1;
for (const auto& [x, nodes] : path) {
if (gcd(nums[node], x) != 1) {
continue;
}
if (nodes.back().second > max_d) {
max_d = nodes.back().second;
result[node] = nodes.back().first;
}
}
} else if (step == 3) {
path[nums[node]].emplace_back(node, depth);
if (adj.count(node)) {
for (const auto& nei : adj.at(node)) {
if (nei == prev) {
continue;
}
stk.emplace_back(1, node, nei, depth + 1);
}
}
} else if (step == 4) {
path[nums[node]].pop_back();
if (empty(path[nums[node]])) {
path.erase(nums[node]);
}
}
}
return result;
}
};
// Time: O(50 * n) = O(n)
// Space: O(n)
class Solution2 {
public:
vector<int> getCoprimes(vector<int>& nums, vector<vector<int>>& edges) {
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]);
}
vector<int> result(size(nums), -1);
unordered_map<int, vector<pair<int, int>>> path;
dfs(nums, adj, -1, 0, 0, &path, &result);
return result;
}
private:
void dfs(const vector<int>& nums, const unordered_map<int, vector<int>>& adj,
int prev, int node, int depth,
unordered_map<int, vector<pair<int, int>>> *path,
vector<int> *result) {
int max_d = -1;
for (const auto& [x, nodes] : *path) {
if (gcd(nums[node], x) != 1) {
continue;
}
if (nodes.back().second > max_d) {
max_d = nodes.back().second;
(*result)[node] = nodes.back().first;
}
}
(*path)[nums[node]].emplace_back(node, depth);
if (adj.count(node)) {
for (const auto& nei : adj.at(node)) {
if (nei == prev) {
continue;
}
dfs(nums, adj, node, nei, depth + 1, path, result);
}
}
(*path)[nums[node]].pop_back();
if (empty((*path)[nums[node]])) {
(*path).erase(nums[node]);
}
}
};