-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathday_08b.cpp
102 lines (92 loc) · 2.86 KB
/
day_08b.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
#include <cmath>
#include <cstdio>
#include <fstream>
#include <iostream>
#include <vector>
#include <queue>
#include <unordered_set>
#include <unordered_map>
#include <algorithm>
#include <chrono>
#include <thread>
#include <memory>
#include <numeric>
using namespace std::chrono_literals;
struct Node {
std::vector<std::unique_ptr<Node>> children;
std::vector<int> metadata;
int sum = -1;
static int calculate_sum_of_nodes(Node* node) {
if (node->sum == -1) {
if ((node->children).empty()) {
node->sum = std::accumulate(std::begin(node->metadata), std::end(node->metadata), 0);
} else {
int sum = 0;
for (const auto m : node->metadata) {
if (m < node->children.size() + 1) {
sum += calculate_sum_of_nodes((node->children[m-1]).get());
}
}
node->sum = sum;
}
}
return node->sum;
}
};
std::unique_ptr<Node> createChild(const std::vector<int>& input, int& index) {
// std::this_thread::sleep_for(1s);
const int n_children = input[index];
const int n_metadata = input[index+1];
// std::cout << "index: " << index << '\n';
// std::cout << "number of children : " << n_children << '\n';
// std::cout << "number of metadata : " << n_metadata << '\n';
index += 2;
std::unique_ptr<Node> this_node = std::make_unique<Node>();
for (int i = 0; i < n_children; i++) {
// std::cout << "index of child: " << index << '\n';
// std::cout << "number of childred of child: " << input[index] << '\n';
this_node->children.push_back(createChild(input, index));
}
for (int i = 0; i < n_metadata; i++) {
// std::cout << "index of metadata: " << index << '\n';
// std::cout << "value of metadata: " << input[index] << '\n';
this_node->metadata.push_back(input[index]);
index++;
}
return this_node;
}
std::unique_ptr<Node> parseTree(const std::string& input) {
size_t start = 0;
const std::string delimiter = " ";
size_t end = input.find(delimiter);
std::vector<int> parsed_input;
while (end < std::string::npos) {
parsed_input.emplace_back(stoi(input.substr(start, end - start)));
start = end + delimiter.size();
end = input.find(delimiter, start);
}
parsed_input.emplace_back(stoi(input.substr(start, input.size() - start)));
int index = 0;
return createChild(parsed_input, index);
}
int sumMetadata(const Node* node) {
int sum = 0;
for (const auto& c : node->children) {
sum += sumMetadata(c.get());
}
sum += std::accumulate(node->metadata.begin(), node->metadata.end(), 0);
return sum;
}
int main(int argc, char* argv[]) {
std::string input = "../input/day_08_input";
if (argc > 1) {
input = argv[1];
}
std::ifstream file(input);
std::string line;
std::getline(file, line);
auto root = parseTree(line);
const int ans = Node::calculate_sum_of_nodes(root.get());
std::cout << ans << '\n';
return ans;
}