Skip to content

Commit 32afc40

Browse files
committedMar 23, 2024
Add 2023 days 20-25 cpp
1 parent efef64d commit 32afc40

19 files changed

+2040
-1
lines changed
 

‎2023/README.md

+6
Original file line numberDiff line numberDiff line change
@@ -27,3 +27,9 @@ This folder contains solutions to each of the problems in Advent of Code 2023 in
2727
| <nobr> [Day 17: Clumsy Crucible](https://adventofcode.com/2023/day/17) </nobr> | <nobr> [Part 1](/2023/cpp/day_17a.cpp) [Part 2](/2023/cpp/day_17b.cpp) </nobr> | </nobr> [Link](/2023/input/day_17_input) </nobr> | </nobr> [Link](/2023/sample_input/day_17_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_17_puzzle) </nobr> |
2828
| <nobr> [Day 18: Lavaduct Lagoon](https://adventofcode.com/2023/day/18) </nobr> | <nobr> [Part 1](/2023/cpp/day_18a.cpp) [Part 2](/2023/cpp/day_18b.cpp) </nobr> | </nobr> [Link](/2023/input/day_18_input) </nobr> | </nobr> [Link](/2023/sample_input/day_18_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_18_puzzle) </nobr> |
2929
| <nobr> [Day 19: Aplenty](https://adventofcode.com/2023/day/19) </nobr> | <nobr> [Part 1](/2023/cpp/day_19a.cpp) [Part 2](/2023/cpp/day_19b.cpp) </nobr> | </nobr> [Link](/2023/input/day_19_input) </nobr> | </nobr> [Link](/2023/sample_input/day_19_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_19_puzzle) </nobr> |
30+
| <nobr> [Day 20: Pulse Propagation](https://adventofcode.com/2023/day/20) </nobr> | <nobr> [Part 1](/2023/cpp/day_20a.cpp) [Part 2](/2023/cpp/day_20b.cpp) </nobr> | </nobr> [Link](/2023/input/day_20_input) </nobr> | </nobr> [Link](/2023/sample_input/day_20_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_20_puzzle) </nobr> |
31+
| <nobr> [Day 21: Step Counter](https://adventofcode.com/2023/day/21) </nobr> | <nobr> [Part 1](/2023/cpp/day_21a.cpp) [Part 2](/2023/cpp/day_21b.cpp) </nobr> | </nobr> [Link](/2023/input/day_21_input) </nobr> | </nobr> [Link](/2023/sample_input/day_21_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_21_puzzle) </nobr> |
32+
| <nobr> [Day 22: Sand Slabs](https://adventofcode.com/2023/day/22) </nobr> | <nobr> [Part 1](/2023/cpp/day_22a.cpp) [Part 2](/2023/cpp/day_22b.cpp) </nobr> | </nobr> [Link](/2023/input/day_22_input) </nobr> | </nobr> [Link](/2023/sample_input/day_22_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_22_puzzle) </nobr> |
33+
| <nobr> [Day 23: A Long Walk](https://adventofcode.com/2023/day/23) </nobr> | <nobr> [Part 1](/2023/cpp/day_23a.cpp) [Part 2](/2023/cpp/day_23b.cpp) </nobr> | </nobr> [Link](/2023/input/day_23_input) </nobr> | </nobr> [Link](/2023/sample_input/day_23_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_23_puzzle) </nobr> |
34+
| <nobr> [Day 24: Never Tell Me The Odds](https://adventofcode.com/2023/day/24) </nobr> | <nobr> [Part 1](/2023/cpp/day_24a.cpp) [Part 2](/2023/cpp/day_24b.cpp) </nobr> | </nobr> [Link](/2023/input/day_24_input) </nobr> | </nobr> [Link](/2023/sample_input/day_24_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_24_puzzle) </nobr> |
35+
| <nobr> [Day 25: Snowverload](https://adventofcode.com/2023/day/25) </nobr> | <nobr> [Part 1](/2023/cpp/day_25a.cpp) | </nobr> [Link](/2023/input/day_25_input) </nobr> | </nobr> [Link](/2023/sample_input/day_25_sample_input) </nobr> | </nobr> [Link](/2023/puzzles/day_25_puzzle) </nobr> |

‎2023/cpp/day_20a.cpp

+207
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,207 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <string>
8+
#include <unordered_set>
9+
#include <vector>
10+
#include <cassert>
11+
#include <memory>
12+
13+
struct Module {
14+
std::string name;
15+
std::vector<std::string> to_modules;
16+
std::vector<std::string> from_modules;
17+
virtual std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) {
18+
return {};
19+
};
20+
virtual void setup() {};
21+
};
22+
23+
struct FF : public Module {
24+
bool state = false;
25+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
26+
// std::cout << __LINE__ << '\n';
27+
28+
std::vector<std::pair<std::string, bool>> return_values;
29+
if (!is_high) {
30+
state = !state;
31+
for (const auto& to : to_modules) {
32+
// std::cout << "TO: " << to << '\n';
33+
return_values.emplace_back(to, state);
34+
}
35+
}
36+
// std::cout << __LINE__ << '\n';
37+
return return_values;
38+
}
39+
};
40+
41+
struct Conjunction : public Module {
42+
std::unordered_map<std::string, bool> state;
43+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
44+
state[source] = is_high;
45+
bool is_low = false;
46+
bool output_value = true;
47+
for (const auto& [name, value] : state) {
48+
// std::cout << name << ' ' << value << '\n';
49+
if (!value) {
50+
is_low = true;
51+
break;
52+
}
53+
}
54+
if(!is_low) output_value = false;
55+
std::vector<std::pair<std::string, bool>> return_values;
56+
for (const auto& to : to_modules) {
57+
// std::cout << "TO: " << to << '\n';
58+
return_values.emplace_back(to, output_value);
59+
}
60+
return return_values;
61+
}
62+
void setup() override {
63+
for (const auto& from : from_modules) {
64+
state[from] = false;
65+
}
66+
}
67+
};
68+
69+
struct Broadcaster : public Module {
70+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
71+
std::vector<std::pair<std::string, bool>> return_values;
72+
for (const auto& to : to_modules) {
73+
// std::cout << "TO: " << to << '\n';
74+
return_values.emplace_back(to, is_high);
75+
}
76+
return return_values;
77+
}
78+
};
79+
80+
std::unique_ptr<Module> create_module (const std::string& name, const std::vector<std::string>& connected_to) {
81+
std::unique_ptr<Module> m_ptr;
82+
if(name[0] == '%') {
83+
FF m;
84+
m.name = name.substr(1, name.size() - 1);
85+
m.to_modules = connected_to;
86+
m_ptr = std::make_unique<FF>(m);
87+
} else if(name[0] == '&') {
88+
Conjunction m;
89+
m.name = name.substr(1, name.size() - 1);
90+
m.to_modules = connected_to;
91+
m_ptr = std::make_unique<Conjunction>(m);
92+
} else if (name == "broadcaster") {
93+
Broadcaster m;
94+
m.name = "broadcaster";
95+
m.to_modules = connected_to;
96+
m_ptr = std::make_unique<Broadcaster>(m);
97+
}
98+
// std::cout << "Created " << m_ptr->name << '\n';
99+
return m_ptr;
100+
}
101+
102+
std::vector<std::string> extract_connected_to(const std::string& s) {
103+
// std::cout << "|" << s << "|" << '\n';
104+
std::vector<std::string> substrs;
105+
106+
std::size_t start = 0;
107+
std::size_t end = s.find(',', start);
108+
while (end != std::string::npos) {
109+
substrs.push_back(s.substr(start, end - start));
110+
start = end + 2;
111+
end = s.find(',', start);
112+
}
113+
substrs.push_back(s.substr(start, s.size() - start));
114+
// std::cout << '|';
115+
// for (const auto s : substrs) {
116+
// std::cout << s << "|";
117+
// }
118+
// std::cout << '\n';
119+
return substrs;
120+
}
121+
122+
std::pair<std::string, std::vector<std::string>> parse (const std::string& line) {
123+
const auto idx = line.find(' ');
124+
return {line.substr(0, idx), extract_connected_to(line.substr(idx+4, line.size() - idx - 4))};
125+
}
126+
127+
struct Pulse {
128+
std::string from;
129+
std::string to;
130+
bool value;
131+
};
132+
133+
int main(int argc, char * argv[]) {
134+
std::string input = "../input/day_20_input";
135+
if (argc > 1) {
136+
input = argv[1];
137+
}
138+
139+
std::string line;
140+
std::fstream file(input);
141+
std::unordered_map<std::string, std::unique_ptr<Module>> modules;
142+
while(std::getline(file, line)) {
143+
const auto [name_with_type, connected_to] = parse (line);
144+
auto tmp = create_module(name_with_type, connected_to);
145+
modules[tmp->name] = std::move(tmp);
146+
// std::cout << __LINE__ << '\n';
147+
}
148+
149+
for (const auto& [name, module] : modules) {
150+
for (const auto& to : module->to_modules) {
151+
if (modules.find(to) == modules.end()) {
152+
Module m;
153+
m.name = to;
154+
modules[to] = std::make_unique<Module>(std::move(m));
155+
}
156+
// std::cout << __LINE__ << '\n';
157+
modules[to]->from_modules.push_back(module->name);
158+
// std::cout << __LINE__ << '\n';
159+
}
160+
}
161+
162+
for (const auto& [name, module] : modules) {
163+
// std::cout << __LINE__ << '\n';
164+
module->setup();
165+
// std::cout << __LINE__ << '\n';
166+
}
167+
168+
std::size_t low_count = 0;
169+
std:;size_t high_count = 0;
170+
for (int i = 0; i < 1000; i++) {
171+
// std::cout << "Press button" << '\n';
172+
std::queue<Pulse> to_process;
173+
{
174+
Pulse p;
175+
p.from = "button";
176+
p.to = "broadcaster";
177+
p.value = false;
178+
to_process.push(p);
179+
}
180+
// std::cout << __LINE__ << '\n';
181+
while (!to_process.empty()) {
182+
const auto pulse = to_process.front();
183+
to_process.pop();
184+
// std::cout << "Processing " << pulse.from <<" -" << (pulse.value ? "high" : "low") << "-> " << pulse.to << '\n';
185+
if(pulse.value) {
186+
high_count++;
187+
} else {
188+
low_count++;
189+
}
190+
// std::cout << __LINE__ << '\n';
191+
const auto modules_to_process = modules[pulse.to]->evaluate(pulse.from, pulse.value);
192+
// std::cout << __LINE__ << '\n';
193+
194+
for (const auto module_to_process : modules_to_process) {
195+
// std::cout << module_to_process.first << '\n';
196+
Pulse new_pulse;
197+
new_pulse.from = pulse.to;
198+
new_pulse.to = module_to_process.first;
199+
new_pulse.value = module_to_process.second;
200+
to_process.push(new_pulse);
201+
}
202+
}
203+
}
204+
// std::cout << high_count << ' ' << low_count << '\n';
205+
std::cout << low_count * high_count << '\n';
206+
return 0;
207+
}

‎2023/cpp/day_20b.cpp

+214
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,214 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <string>
8+
#include <unordered_set>
9+
#include <vector>
10+
#include <cassert>
11+
#include <memory>
12+
13+
struct Module {
14+
std::string name;
15+
std::vector<std::string> to_modules;
16+
std::vector<std::string> from_modules;
17+
virtual std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) {
18+
return {};
19+
};
20+
virtual void setup() {};
21+
};
22+
23+
struct FF : public Module {
24+
bool state = false;
25+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
26+
std::vector<std::pair<std::string, bool>> return_values;
27+
if (!is_high) {
28+
state = !state;
29+
for (const auto& to : to_modules) {
30+
// std::cout << "TO: " << to << '\n';
31+
return_values.emplace_back(to, state);
32+
}
33+
}
34+
return return_values;
35+
}
36+
};
37+
38+
struct Conjunction : public Module {
39+
std::unordered_map<std::string, bool> state;
40+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
41+
state[source] = is_high;
42+
bool is_low = false;
43+
bool output_value = true;
44+
for (const auto& [name, value] : state) {
45+
if (!value) {
46+
is_low = true;
47+
break;
48+
}
49+
}
50+
if(!is_low) output_value = false;
51+
std::vector<std::pair<std::string, bool>> return_values;
52+
for (const auto& to : to_modules) {
53+
// std::cout << "TO: " << to << '\n';
54+
return_values.emplace_back(to, output_value);
55+
}
56+
return return_values;
57+
}
58+
void setup() override {
59+
for (const auto& from : from_modules) {
60+
state[from] = false;
61+
}
62+
}
63+
};
64+
65+
struct Broadcaster : public Module {
66+
std::vector<std::pair<std::string, bool>> evaluate(const std::string& source, const bool is_high) override {
67+
std::vector<std::pair<std::string, bool>> return_values;
68+
for (const auto& to : to_modules) {
69+
// std::cout << "TO: " << to << '\n';
70+
return_values.emplace_back(to, is_high);
71+
}
72+
return return_values;
73+
}
74+
};
75+
76+
std::unique_ptr<Module> create_module (const std::string& name, const std::vector<std::string>& connected_to) {
77+
std::unique_ptr<Module> m_ptr;
78+
if(name[0] == '%') {
79+
FF m;
80+
m.name = name.substr(1, name.size() - 1);
81+
m.to_modules = connected_to;
82+
m_ptr = std::make_unique<FF>(m);
83+
} else if(name[0] == '&') {
84+
Conjunction m;
85+
m.name = name.substr(1, name.size() - 1);
86+
m.to_modules = connected_to;
87+
m_ptr = std::make_unique<Conjunction>(m);
88+
} else if (name == "broadcaster") {
89+
Broadcaster m;
90+
m.name = "broadcaster";
91+
m.to_modules = connected_to;
92+
m_ptr = std::make_unique<Broadcaster>(m);
93+
}
94+
return m_ptr;
95+
}
96+
97+
std::vector<std::string> extract_connected_to(const std::string& s) {
98+
std::vector<std::string> substrs;
99+
100+
std::size_t start = 0;
101+
std::size_t end = s.find(',', start);
102+
while (end != std::string::npos) {
103+
substrs.push_back(s.substr(start, end - start));
104+
start = end + 2;
105+
end = s.find(',', start);
106+
}
107+
substrs.push_back(s.substr(start, s.size() - start));
108+
return substrs;
109+
}
110+
111+
std::pair<std::string, std::vector<std::string>> parse (const std::string& line) {
112+
const auto idx = line.find(' ');
113+
return {line.substr(0, idx), extract_connected_to(line.substr(idx+4, line.size() - idx - 4))};
114+
}
115+
116+
struct Pulse {
117+
std::string from;
118+
std::string to;
119+
bool value;
120+
};
121+
122+
int main(int argc, char * argv[]) {
123+
std::string input = "../input/day_20_input";
124+
if (argc > 1) {
125+
input = argv[1];
126+
}
127+
128+
std::string line;
129+
std::fstream file(input);
130+
std::unordered_map<std::string, std::unique_ptr<Module>> modules;
131+
while(std::getline(file, line)) {
132+
const auto [name_with_type, connected_to] = parse (line);
133+
auto tmp = create_module(name_with_type, connected_to);
134+
modules[tmp->name] = std::move(tmp);
135+
// std::cout << __LINE__ << '\n';
136+
}
137+
138+
for (const auto& [name, module] : modules) {
139+
for (const auto& to : module->to_modules) {
140+
if (modules.find(to) == modules.end()) {
141+
Module m;
142+
m.name = to;
143+
modules[to] = std::make_unique<Module>(std::move(m));
144+
}
145+
// std::cout << __LINE__ << '\n';
146+
modules[to]->from_modules.push_back(module->name);
147+
// std::cout << __LINE__ << '\n';
148+
}
149+
}
150+
151+
std::unordered_map<std::string, std::size_t> periods;
152+
// By inspection it can be seen that rx needs to receive a single pulse from bb (the only module it is connected to)
153+
// Hence bb needs to receive and remember high pulses from the modules it is connected to
154+
// Debug: bb is connected to:
155+
for (const auto& ele : modules[modules["rx"]->from_modules[0]]->from_modules ) {
156+
// std::cout << ele << " | ";
157+
periods[ele] = 0;
158+
}
159+
// std::cout << '\n';
160+
for (const auto& [name, module] : modules) {
161+
module->setup();
162+
}
163+
164+
std::size_t low_count = 0;
165+
std::size_t high_count = 0;
166+
int count = 0;
167+
for (int i = 0; i < 100000000; i++) {
168+
// std::cout << "Press button" << '\n';
169+
std::queue<Pulse> to_process;
170+
{
171+
Pulse p;
172+
p.from = "button";
173+
p.to = "broadcaster";
174+
p.value = false;
175+
to_process.push(p);
176+
}
177+
while (!to_process.empty()) {
178+
const auto pulse = to_process.front();
179+
to_process.pop();
180+
// std::cout << "Processing " << pulse.from <<" -" << (pulse.value ? "high" : "low") << "-> " << pulse.to << '\n';
181+
if(pulse.value) {
182+
high_count++;
183+
} else {
184+
low_count++;
185+
}
186+
if (pulse.value && (periods.find(pulse.from) != periods.end())) {
187+
if (periods[pulse.from] == 0) {
188+
periods[pulse.from] = i;
189+
count++;
190+
}
191+
if (count == 4) {
192+
std::size_t ans = 1;
193+
for (auto& [name, val] : periods) {
194+
val+=1; // Since pulse count (i) started from 0
195+
ans = std::lcm(ans, val);
196+
}
197+
std::cout << ans << '\n';
198+
return 0;
199+
}
200+
}
201+
202+
const auto modules_to_process = modules[pulse.to]->evaluate(pulse.from, pulse.value);
203+
for (const auto module_to_process : modules_to_process) {
204+
// std::cout << module_to_process.first << '\n';
205+
Pulse new_pulse;
206+
new_pulse.from = pulse.to;
207+
new_pulse.to = module_to_process.first;
208+
new_pulse.value = module_to_process.second;
209+
to_process.push(new_pulse);
210+
}
211+
}
212+
}
213+
return 0;
214+
}

‎2023/cpp/day_21a.cpp

+103
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,103 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <string>
8+
#include <unordered_set>
9+
#include <vector>
10+
#include <cassert>
11+
#include <memory>
12+
13+
constexpr int n = 64;
14+
15+
struct PointState {
16+
int row;
17+
int col;
18+
int step;
19+
20+
PointState(const int row = 0, const int col = 0, const int step = 0) : row(row), col(col), step(step) {}
21+
22+
PointState operator + (const PointState& p) const {
23+
return PointState (p.row + row, p.col + col, p.step + step);
24+
}
25+
26+
bool operator == (const PointState& p) const {
27+
return p.row == row && p.col == col && p.step == step;
28+
}
29+
};
30+
31+
struct hasher {
32+
std::size_t operator() (const PointState& ps) const {
33+
return ps.row + ps.col;
34+
}
35+
};
36+
37+
const std::vector<PointState> motions {
38+
PointState(-1,0,1),
39+
PointState(0,1,1),
40+
PointState(1,0,1),
41+
PointState(0, -1,1),
42+
};
43+
44+
int main(int argc, char * argv[]) {
45+
std::string input = "../input/day_21_input";
46+
if (argc > 1) {
47+
input = argv[1];
48+
}
49+
50+
std::string line;
51+
std::fstream file(input);
52+
std::vector<std::string> map;
53+
int step = 0;
54+
PointState S;
55+
while(std::getline(file, line)) {
56+
map.push_back(line);
57+
for (int i = 0; i < line.size(); i++) {
58+
if (line[i] == 'S') {
59+
S.row = map.size() - 1;
60+
S.col = i;
61+
S.step = 0;
62+
}
63+
}
64+
}
65+
std::queue<PointState> queue;
66+
std::unordered_set<PointState, hasher> seen;
67+
std::unordered_set<PointState, hasher> end_states;
68+
queue.push(S);
69+
while(!queue.empty()) {
70+
const auto current = queue.front();
71+
queue.pop();
72+
if (seen.find(current) != seen.end()) {
73+
continue;
74+
}
75+
if (current.step > n) continue;
76+
if (current.step == n) end_states.insert(current);
77+
seen.insert(current);
78+
for (const auto& motion : motions) {
79+
const auto new_state = current + motion;
80+
if (new_state.row >= map.size() || new_state.row < 0 || new_state.col < 0 || new_state.col >= map[0].size()) {
81+
continue;
82+
}
83+
if (map[new_state.row][new_state.col] == '#') {
84+
continue;
85+
}
86+
queue.push(new_state);
87+
}
88+
}
89+
for (const auto & end_state : end_states) {
90+
map[end_state.row][end_state.col] = 'O';
91+
}
92+
int count = 0;
93+
for (const auto& row : map) {
94+
std::cout << row << '\n';
95+
for (const auto c : row) {
96+
if (c == 'O') {
97+
count++;
98+
}
99+
}
100+
}
101+
std::cout << count << '\n';
102+
return 0;
103+
}

‎2023/cpp/day_21b.cpp

+203
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,203 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <string>
8+
#include <unordered_set>
9+
#include <vector>
10+
#include <cassert>
11+
#include <memory>
12+
13+
// This will work with both the sample input and the main input
14+
// The core idea here is that the edges of the square grid are always empty
15+
// This leads to a highway that can be used to reach grids
16+
// further away from the starting point
17+
// At some unknown distance it becomes more efficient to
18+
// 1. move from the starting point to the highway
19+
// 2. traverse the highway to the target point's grid
20+
// 3. move from the highway to the target point
21+
// instead of going through the grids between the start and end point
22+
// Let this distance cover `heuristic_1` grids
23+
// So up until that heuristic brute force is used to find the number of reachable points.
24+
//
25+
// The infinite grid can be represented as
26+
// ^^^^^
27+
// |||||
28+
// <-CEEEC->
29+
// <-E...E->
30+
// <-E...E->
31+
// <-E...E->
32+
// <-CEEEC->
33+
// |||||
34+
// vvvvv
35+
// where the `.`s are bruteforced
36+
// the edges `E` represent all the the points in the infinite grid as represented by the arrows next to them
37+
// the corners `C` represent all the the points in the infinite grid as represented and enclosed by the arrows next to them
38+
//
39+
// Reference:
40+
// https://github.com/jonathanpaulson/AdventOfCode/blob/master/2023/21.py
41+
42+
constexpr int heuristic_1 = 4;
43+
constexpr std::size_t n_steps = 26501365;
44+
45+
struct Point {
46+
int tile_row;
47+
int tile_col;
48+
int row;
49+
int col;
50+
std::size_t step;
51+
std::size_t hash;
52+
53+
Point(const int tile_row = 0, const int tile_col = 0, const int row = 0, const int col = 0, const int step = 0) : tile_row(tile_row), tile_col(tile_col), row(row), col(col), step(step) {
54+
hash = tile_row * tile_col * 10000 + row * col;
55+
}
56+
57+
Point operator + (const Point& p) const {
58+
return Point (p.tile_row + tile_row, p.tile_col + tile_col, p.row + row, p.col + col, p.step + step);
59+
}
60+
61+
bool operator == (const Point& p) const {
62+
return p.tile_row == tile_row && p.tile_col == tile_col && p.row == row && p.col == col && p.step == step;
63+
}
64+
void update_hash() {
65+
hash = tile_row * tile_col * 10000 + row * col;
66+
}
67+
};
68+
69+
struct hasher {
70+
std::size_t operator() (const Point& ps) const {
71+
return ps.hash;
72+
}
73+
};
74+
75+
const std::vector<Point> motions {
76+
Point(0,0,-1,0,0),
77+
Point(0,0,0,1,0),
78+
Point(0,0,1,0,0),
79+
Point(0,0,0,-1,0),
80+
};
81+
82+
// Should replace point with another struct here
83+
std::unordered_map<Point, std::size_t, hasher> temp_map;
84+
std::size_t helper(const std::size_t& step, const int increment, const std::size_t& n_rows) {
85+
std::size_t temp = (n_steps >= step) ? (n_steps - step) / n_rows : 0;
86+
// std::cout << temp << '\n';
87+
if (temp_map.find(Point(step, increment,0, 0, 0)) != temp_map.end()) {
88+
return temp_map[Point(step, increment, 0, 0, 0)];
89+
}
90+
std::size_t return_val = 0;
91+
for (std::size_t idx = 1; idx <= temp; idx++) {
92+
if (((step + n_rows * idx) <= n_steps) && ((step + n_rows * idx) % 2 == (n_steps % 2))) {
93+
return_val += (increment == 2) ? (idx + 1) : 1;
94+
}
95+
}
96+
temp_map[Point(step, increment, 0, 0, 0)] = return_val;
97+
// std::cout << return_val << '\n';
98+
return return_val;
99+
}
100+
101+
std::unordered_map<Point, std::size_t, hasher> create_count_map(const std::vector<std::string>& map, const Point& S) {
102+
std::queue<Point> queue;
103+
std::unordered_map<Point, std::size_t, hasher> count_map;
104+
const auto row_size = map.size();
105+
const auto col_size = map[0].size();
106+
// std::cout << row_size << ' ' << col_size << '\n';
107+
// exit(0);
108+
queue.push(S);
109+
// std::cout << S.row << ' ' << S.col << '\n';
110+
while(!queue.empty()) {
111+
auto current = queue.front();
112+
queue.pop();
113+
// std::cout << current.tile_row << ' '
114+
// << current.tile_col << ' '
115+
// << current.row << ' '
116+
// << current.col << ' '
117+
// << current.step << '\n';
118+
if (current.row < 0) {
119+
current.tile_row -= 1;
120+
current.row += row_size;
121+
}
122+
if (current.row >= row_size) {
123+
current.tile_row += 1;
124+
current.row -= row_size;
125+
}
126+
if (current.col < 0) {
127+
current.tile_col -= 1;
128+
current.col += col_size;
129+
}
130+
if (current.col >= col_size) {
131+
current.tile_col += 1;
132+
current.col -= col_size;
133+
}
134+
// current.update_hash();
135+
if (current.row < 0 || current.row >= row_size || current.col < 0 || current.col >= col_size || map[current.row][current.col] == '#') continue;
136+
if (count_map.find(Point(current.tile_row, current.tile_col, current.row, current.col, 0)) != count_map.end()) continue;
137+
if (std::abs(current.tile_row) > heuristic_1 || std::abs(current.tile_col) > heuristic_1) continue;
138+
count_map[Point(current.tile_row, current.tile_col, current.row, current.col, 0)] = current.step;
139+
for (const auto& motion : motions) {
140+
queue.emplace(current.tile_row, current.tile_col, current.row + motion.row, current.col + motion.col, current.step + 1);
141+
}
142+
}
143+
return count_map;
144+
}
145+
146+
int main(int argc, char * argv[]) {
147+
std::string input = "../input/day_21_input";
148+
if (argc > 1) {
149+
input = argv[1];
150+
}
151+
152+
std::string line;
153+
std::fstream file(input);
154+
Point S(0,0,0,0,0);
155+
std::vector<std::string> map;
156+
while(std::getline(file, line)) {
157+
if (line.empty()) continue;
158+
map.push_back(line);
159+
for (int i = 0; i < line.size(); i++) {
160+
if (line[i] == 'S') {
161+
S.row = map.size() - 1;
162+
S.col = i;
163+
S.step = 0;
164+
}
165+
}
166+
}
167+
168+
// Count map already contains the tiles in the square contained by
169+
// (-heuristic_1, -heuristic_1) to (heuristic_1, 1heuristic_1)
170+
// inclusive of the border.
171+
std::vector<int> OPT;
172+
for (int i = -heuristic_1; i <= heuristic_1; i++){
173+
OPT.push_back(i);
174+
}
175+
const auto count_map = create_count_map(map, S);
176+
std::size_t ans = 0;
177+
const auto n_rows = map.size();
178+
const auto n_cols = map[0].size();
179+
for (int row = 0; row < n_rows; row++) {
180+
for (int col = 0; col < n_cols; col++) {
181+
const auto p = Point(0, 0, row, col, 0);
182+
if (count_map.find(p) != count_map.end()) {
183+
for (const auto tile_row : OPT) {
184+
for (const auto tile_col : OPT) {
185+
const auto step = count_map.at(Point(tile_row, tile_col, row, col, 0));
186+
// std::cout <<tile_row <<' ' << tile_col<<' ' << row<<' ' << col<<' ' << 0 << '\n';
187+
if ((step % 2) == (n_steps % 2) && step <= n_steps) {
188+
ans += 1;
189+
}
190+
if ((tile_row == -heuristic_1 || tile_row == heuristic_1) && (tile_col == -heuristic_1 || tile_col == heuristic_1)) {
191+
ans += helper (step, 2, n_rows);
192+
}
193+
else if ((tile_row == -heuristic_1 || tile_row == heuristic_1) || (tile_col == -heuristic_1 || tile_col == heuristic_1)) {
194+
ans += helper (step, 1, n_rows);
195+
}
196+
}
197+
}
198+
}
199+
}
200+
}
201+
std::cout << ans << '\n';
202+
return 0;
203+
}

‎2023/cpp/day_22a.cpp

+175
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,175 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <stack>
8+
#include <string>
9+
#include <unordered_set>
10+
#include <vector>
11+
#include <regex>
12+
#include <cassert>
13+
14+
15+
struct Brick {
16+
std::array<int, 3> start {0,0,0};
17+
std::array<int, 3> end {0,0,0};
18+
int id = 0;
19+
bool operator == (const Brick& b) const {
20+
return b.start == start && b.end == end;
21+
}
22+
};
23+
24+
bool check_if_overlap(const Brick& b1, const Brick& b2) {
25+
bool x_overlap = false;
26+
bool y_overlap = false;
27+
x_overlap = std::min(b2.start[0], b2.end[0]) <= b1.start[0] && b1.start[0] <= std::max(b2.start[0], b2.end[0]);
28+
y_overlap = std::min(b2.start[1], b2.end[1]) <= b1.start[1] && b1.start[1] <= std::max(b2.start[1], b2.end[1]);
29+
if (x_overlap && y_overlap) return true;
30+
x_overlap = x_overlap || std::min(b2.start[0], b2.end[0]) <= b1.end[0] && b1.end[0] <= std::max(b2.start[0], b2.end[0]);
31+
y_overlap = y_overlap || std::min(b2.start[1], b2.end[1]) <= b1.end[1] && b1.end[1] <= std::max(b2.start[1], b2.end[1]);
32+
if (x_overlap && y_overlap) return true;
33+
x_overlap = x_overlap || std::min(b1.start[0], b1.end[0]) <= b2.start[0] && b2.start[0] <= std::max(b1.start[0], b1.end[0]);
34+
y_overlap = y_overlap || std::min(b1.start[1], b1.end[1]) <= b2.start[1] && b2.start[1] <= std::max(b1.start[1], b1.end[1]);
35+
if (x_overlap && y_overlap) return true;
36+
x_overlap = x_overlap || std::min(b1.start[0], b1.end[0]) <= b2.end[0] && b2.end[0] <= std::max(b1.start[0], b1.end[0]);
37+
y_overlap = y_overlap || std::min(b1.start[1], b1.end[1]) <= b2.end[1] && b2.end[1] <= std::max(b1.start[1], b1.end[1]);
38+
if (x_overlap && y_overlap) return true;
39+
return false;
40+
}
41+
42+
int main(int argc, char * argv[]) {
43+
std::string input = "../input/day_22_input";
44+
if (argc > 1) {
45+
input = argv[1];
46+
}
47+
48+
std::string line;
49+
std::fstream file(input);
50+
const std::regex mask_pattern(R"((-?[0-9]+),(-?[0-9]+),(-?[0-9]+)~(-?[0-9]+),(-?[0-9]+),(-?[0-9]+))");
51+
std::smatch mask_match;
52+
std::vector<Brick> bricks;
53+
while(std::getline(file, line)) {
54+
std::regex_search(line, mask_match, mask_pattern);
55+
Brick b;
56+
b.id = bricks.size();
57+
b.start[0] = std::stoi(mask_match[1]);
58+
b.start[1] = std::stoi(mask_match[2]);
59+
b.start[2] = std::stoi(mask_match[3]);
60+
b.end[0] = std::stoi(mask_match[4]);
61+
b.end[1] = std::stoi(mask_match[5]);
62+
b.end[2] = std::stoi(mask_match[6]);
63+
bricks.push_back(b);
64+
// std::cout << b.start[0] << ' ' << b.start[1] << ' '
65+
// << b.start[2] << ' ' << b.end[0] << ' '
66+
// << b.end[1]<< ' ' << b.end[2] << '\n';
67+
}
68+
for (auto& b : bricks) {
69+
if (b.start[2] > b.end[2]) {
70+
std::swap(b.start, b.end);
71+
}
72+
}
73+
std::sort(
74+
std::begin(bricks),
75+
std::end(bricks),
76+
[](const auto& b1, const auto& b2) {
77+
return b1.start[2] < b2.start[2];
78+
}
79+
);
80+
{
81+
const auto h = bricks[0].end[2] - bricks[0].start[2];
82+
bricks[0].start[2] = 1;
83+
bricks[0].end[2] = 1 + h;
84+
}
85+
for (int i = 1; i < bricks.size(); i++) {
86+
// std::cout << "\nBrick moving down: " << char('A' + bricks[i].id) << '\n';
87+
const auto h = bricks[i].end[2] - bricks[i].start[2];
88+
for (int j = i - 1; j >= 0; j--) {
89+
// std::cout << "Comparing with: " << char('A' + bricks[j].id) << '\n';
90+
bricks[i].start[2] = bricks[j].start[2];
91+
bricks[i].end[2] = bricks[i].start[2] + h;
92+
if (check_if_overlap(bricks[i], bricks[j])) {
93+
// std::cout << "Overlap found between " << char('A' + bricks[i].id) << " and " << char('A' + bricks[j].id) << '\n';
94+
// std::cout << char('A' + bricks[j].id) << " occupies " << bricks[j].start[2] << "->" << bricks[j].end[2] << " in Z" << '\n';
95+
bricks[i].start[2] = bricks[j].end[2] + 1;
96+
bricks[i].end[2] = bricks[i].start[2] + h;
97+
// std::cout << char('A' + bricks[i].id) << " now set to " << bricks[i].start[2] << "->" << bricks[i].end[2] << '\n';
98+
break;
99+
}
100+
}
101+
std::partial_sort(
102+
std::begin(bricks),
103+
std::next(std::begin(bricks), i),
104+
std::end(bricks),
105+
[](const auto& b1, const auto& b2) {
106+
return b1.end[2] < b2.end[2];
107+
}
108+
);
109+
}
110+
std::array<int, 3> values {0,0,0};
111+
for (const auto& b : bricks) {
112+
values[0] = std::max(values[0], b.start[0]);
113+
values[0] = std::max(values[0], b.end[0]);
114+
values[1] = std::max(values[1], b.start[1]);
115+
values[1] = std::max(values[1], b.end[1]);
116+
values[2] = std::max(values[2], b.start[2]);
117+
values[2] = std::max(values[2], b.end[2]);
118+
}
119+
std::vector<std::vector<std::vector<int>>> map (
120+
values[0]+1, std::vector<std::vector<int>>(
121+
values[1]+1, std::vector<int>(
122+
values[2]+1, 0
123+
)
124+
)
125+
);
126+
// std::cout << values[0] << ' ' << values[1] << ' ' << values[2] << '\n';
127+
for (const auto& b : bricks) {
128+
for (int x = std::min(b.start[0],b.end[0]); x <= std::max(b.start[0],b.end[0]); x++) {
129+
for (int y = std::min(b.start[1], b.end[1]); y <= std::max(b.start[1],b.end[1]); y++) {
130+
for (int z = std::min(b.start[2], b.end[2]); z <= std::max(b.start[2],b.end[2]); z++) {
131+
// std::cout << x << ' ' << y << ' ' << z << '\n';
132+
map[x][y][z] = b.id;
133+
}
134+
}
135+
}
136+
}
137+
138+
// for (int z = 0; z <= values[2]; z++) {
139+
// for (int x = 0; x <= values[0]; x++) {
140+
// std::cout << map[x][0][z];
141+
// }
142+
// std::cout << '\n';
143+
// }
144+
145+
std::unordered_map<int, std::vector<int>> supports;
146+
std::unordered_map<int, std::vector<int>> supported_by;
147+
for(const auto& b : bricks) {
148+
supports[b.id] = {};
149+
supported_by[b.id] = {};
150+
}
151+
for (int i = 0; i < bricks.size(); i++) {
152+
for (int j = 0; j < bricks.size(); j++) {
153+
if (i == j) continue;
154+
if(bricks[i].end[2] + 1 == bricks[j].start[2] && check_if_overlap(bricks[i], bricks[j])) {
155+
supports[bricks[i].id].push_back(bricks[j].id);
156+
supported_by[bricks[j].id].push_back(bricks[i].id);
157+
}
158+
}
159+
}
160+
int count = 0;
161+
for (const auto& [current, supported_by_current] : supports) {
162+
bool can_be_disintegrated = true;
163+
for (const auto& ele : supported_by_current) {
164+
if (supported_by[ele].size() == 1) {
165+
can_be_disintegrated = false;
166+
break;
167+
}
168+
}
169+
if (can_be_disintegrated) {
170+
count++;
171+
}
172+
}
173+
std::cout << count << '\n';
174+
return 0;
175+
}

‎2023/cpp/day_22b.cpp

+194
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,194 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <stack>
8+
#include <string>
9+
#include <unordered_set>
10+
#include <vector>
11+
#include <regex>
12+
#include <cassert>
13+
14+
15+
struct Brick {
16+
std::array<int, 3> start {0,0,0};
17+
std::array<int, 3> end {0,0,0};
18+
int id = 0;
19+
bool operator == (const Brick& b) const {
20+
return b.start == start && b.end == end;
21+
}
22+
};
23+
24+
bool check_if_overlap(const Brick& b1, const Brick& b2) {
25+
bool x_overlap = false;
26+
bool y_overlap = false;
27+
x_overlap = std::min(b2.start[0], b2.end[0]) <= b1.start[0] && b1.start[0] <= std::max(b2.start[0], b2.end[0]);
28+
y_overlap = std::min(b2.start[1], b2.end[1]) <= b1.start[1] && b1.start[1] <= std::max(b2.start[1], b2.end[1]);
29+
if (x_overlap && y_overlap) return true;
30+
x_overlap = x_overlap || std::min(b2.start[0], b2.end[0]) <= b1.end[0] && b1.end[0] <= std::max(b2.start[0], b2.end[0]);
31+
y_overlap = y_overlap || std::min(b2.start[1], b2.end[1]) <= b1.end[1] && b1.end[1] <= std::max(b2.start[1], b2.end[1]);
32+
if (x_overlap && y_overlap) return true;
33+
x_overlap = x_overlap || std::min(b1.start[0], b1.end[0]) <= b2.start[0] && b2.start[0] <= std::max(b1.start[0], b1.end[0]);
34+
y_overlap = y_overlap || std::min(b1.start[1], b1.end[1]) <= b2.start[1] && b2.start[1] <= std::max(b1.start[1], b1.end[1]);
35+
if (x_overlap && y_overlap) return true;
36+
x_overlap = x_overlap || std::min(b1.start[0], b1.end[0]) <= b2.end[0] && b2.end[0] <= std::max(b1.start[0], b1.end[0]);
37+
y_overlap = y_overlap || std::min(b1.start[1], b1.end[1]) <= b2.end[1] && b2.end[1] <= std::max(b1.start[1], b1.end[1]);
38+
if (x_overlap && y_overlap) return true;
39+
return false;
40+
}
41+
42+
int main(int argc, char * argv[]) {
43+
std::string input = "../input/day_22_input";
44+
if (argc > 1) {
45+
input = argv[1];
46+
}
47+
48+
std::string line;
49+
std::fstream file(input);
50+
const std::regex mask_pattern(R"((-?[0-9]+),(-?[0-9]+),(-?[0-9]+)~(-?[0-9]+),(-?[0-9]+),(-?[0-9]+))");
51+
std::smatch mask_match;
52+
std::vector<Brick> bricks;
53+
while(std::getline(file, line)) {
54+
std::regex_search(line, mask_match, mask_pattern);
55+
Brick b;
56+
b.id = bricks.size();
57+
b.start[0] = std::stoi(mask_match[1]);
58+
b.start[1] = std::stoi(mask_match[2]);
59+
b.start[2] = std::stoi(mask_match[3]);
60+
b.end[0] = std::stoi(mask_match[4]);
61+
b.end[1] = std::stoi(mask_match[5]);
62+
b.end[2] = std::stoi(mask_match[6]);
63+
bricks.push_back(b);
64+
// std::cout << b.start[0] << ' ' << b.start[1] << ' '
65+
// << b.start[2] << ' ' << b.end[0] << ' '
66+
// << b.end[1]<< ' ' << b.end[2] << '\n';
67+
}
68+
for (auto& b : bricks) {
69+
if (b.start[2] > b.end[2]) {
70+
std::swap(b.start, b.end);
71+
}
72+
}
73+
std::sort(
74+
std::begin(bricks),
75+
std::end(bricks),
76+
[](const auto& b1, const auto& b2) {
77+
return b1.start[2] < b2.start[2];
78+
}
79+
);
80+
{
81+
const auto h = bricks[0].end[2] - bricks[0].start[2];
82+
bricks[0].start[2] = 1;
83+
bricks[0].end[2] = 1 + h;
84+
}
85+
for (int i = 1; i < bricks.size(); i++) {
86+
// std::cout << "\nBrick moving down: " << char('A' + bricks[i].id) << '\n';
87+
const auto h = bricks[i].end[2] - bricks[i].start[2];
88+
for (int j = i - 1; j >= 0; j--) {
89+
// std::cout << "Comparing with: " << char('A' + bricks[j].id) << '\n';
90+
bricks[i].start[2] = bricks[j].start[2];
91+
bricks[i].end[2] = bricks[i].start[2] + h;
92+
if (check_if_overlap(bricks[i], bricks[j])) {
93+
// std::cout << "Overlap found between " << char('A' + bricks[i].id) << " and " << char('A' + bricks[j].id) << '\n';
94+
// std::cout << char('A' + bricks[j].id) << " occupies " << bricks[j].start[2] << "->" << bricks[j].end[2] << " in Z" << '\n';
95+
bricks[i].start[2] = bricks[j].end[2] + 1;
96+
bricks[i].end[2] = bricks[i].start[2] + h;
97+
// std::cout << char('A' + bricks[i].id) << " now set to " << bricks[i].start[2] << "->" << bricks[i].end[2] << '\n';
98+
break;
99+
}
100+
}
101+
// While the bricks are arranged in z bfore they start falling,
102+
// they might have a differnt height order after they land
103+
std::partial_sort(
104+
std::begin(bricks),
105+
std::next(std::begin(bricks), i),
106+
std::end(bricks),
107+
[](const auto& b1, const auto& b2) {
108+
return b1.end[2] < b2.end[2];
109+
}
110+
);
111+
}
112+
std::array<int, 3> values {0,0,0};
113+
for (const auto& b : bricks) {
114+
values[0] = std::max(values[0], b.start[0]);
115+
values[0] = std::max(values[0], b.end[0]);
116+
values[1] = std::max(values[1], b.start[1]);
117+
values[1] = std::max(values[1], b.end[1]);
118+
values[2] = std::max(values[2], b.start[2]);
119+
values[2] = std::max(values[2], b.end[2]);
120+
}
121+
std::vector<std::vector<std::vector<int>>> map (
122+
values[0]+1, std::vector<std::vector<int>>(
123+
values[1]+1, std::vector<int>(
124+
values[2]+1, 0
125+
)
126+
)
127+
);
128+
// std::cout << values[0] << ' ' << values[1] << ' ' << values[2] << '\n';
129+
for (const auto& b : bricks) {
130+
for (int x = std::min(b.start[0],b.end[0]); x <= std::max(b.start[0],b.end[0]); x++) {
131+
for (int y = std::min(b.start[1], b.end[1]); y <= std::max(b.start[1],b.end[1]); y++) {
132+
for (int z = std::min(b.start[2], b.end[2]); z <= std::max(b.start[2],b.end[2]); z++) {
133+
// std::cout << x << ' ' << y << ' ' << z << '\n';
134+
map[x][y][z] = b.id;
135+
}
136+
}
137+
}
138+
}
139+
140+
// for (int z = 0; z <= values[2]; z++) {
141+
// for (int x = 0; x <= values[0]; x++) {
142+
// std::cout << map[x][0][z];
143+
// }
144+
// std::cout << '\n';
145+
// }
146+
147+
std::unordered_map<int, std::vector<int>> supports;
148+
std::unordered_map<int, std::vector<int>> supported_by;
149+
for(const auto& b : bricks) {
150+
supports[b.id] = {};
151+
supported_by[b.id] = {};
152+
}
153+
for (int i = 0; i < bricks.size(); i++) {
154+
for (int j = 0; j < bricks.size(); j++) {
155+
if (i == j) continue;
156+
if(bricks[i].end[2] + 1 == bricks[j].start[2] && check_if_overlap(bricks[i], bricks[j])) {
157+
supports[bricks[i].id].push_back(bricks[j].id);
158+
supported_by[bricks[j].id].push_back(bricks[i].id);
159+
}
160+
}
161+
}
162+
std::size_t count = 0;
163+
for (const auto& b : bricks) {
164+
std::queue<int> q;
165+
for (const auto& ele : supports[b.id]) {
166+
q.push(ele);
167+
}
168+
// Do not cache bricks that might have been seen as
169+
// they are added each time one of the bricks that supports them is destroyed
170+
// Only the last time they are added,
171+
// at which point all the bricks supporting them might have been destroyed
172+
// will they be free fall and hence be counted
173+
std::unordered_set<int> moved {b.id};
174+
while(!q.empty()) {
175+
const auto current = q.front();
176+
q.pop();
177+
bool will_move = true;
178+
for (const auto& ele : supported_by[current]) {
179+
if (moved.find(ele) == moved.end()) {
180+
will_move = false;
181+
break;
182+
}
183+
}
184+
if (!will_move) continue;
185+
moved.insert(current);
186+
for (const auto& supported_by_current : supports[current]) {
187+
q.push(supported_by_current);
188+
}
189+
}
190+
count += moved.size() - 1;
191+
}
192+
std::cout << count << '\n';
193+
return 0;
194+
}

‎2023/cpp/day_23a.cpp

+133
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,133 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <stack>
8+
#include <string>
9+
#include <unordered_set>
10+
#include <vector>
11+
12+
struct Point {
13+
int row;
14+
int col;
15+
16+
Point(const int row = 0, const int col = 0) : row(row), col(col) {}
17+
18+
Point operator + (const Point& p) const {
19+
return Point (p.row + row, p.col + col);
20+
}
21+
22+
bool operator == (const Point& p) const {
23+
return p.row == row && p.col == col;
24+
}
25+
};
26+
27+
struct hasher {
28+
std::size_t operator() (const Point& p) const {
29+
return p.row + p.col;
30+
}
31+
};
32+
33+
const std::vector<Point> motions ={
34+
Point(-1,0),
35+
Point(0,1),
36+
Point(1,0),
37+
Point(0, -1),
38+
};
39+
40+
bool in_limits(const std::vector<std::string>& map, const int row, const int col) {
41+
return row >= 0 && row < map.size() && col >= 0 && col < map[0].size();
42+
}
43+
44+
void dfs(std::stack<Point>& stack, const std::vector<std::string>& map, int steps, int& max_steps, const Point& dest, std::vector<std::vector<bool>>& seen) {
45+
if (!stack.empty()) {
46+
const Point current = stack.top();
47+
seen[current.row][current.col] = true;
48+
if (current == dest) {
49+
max_steps = std::max(steps, max_steps);
50+
seen[current.row][current.col] = false;
51+
return;
52+
}
53+
if (map[current.row][current.col] != '.') {
54+
Point next;
55+
if (map[current.row][current.col] == '^') {
56+
next.row = current.row - 1;
57+
next.col = current.col;
58+
}
59+
else if (map[current.row][current.col] == '>') {
60+
next.row = current.row;
61+
next.col = current.col + 1;
62+
}
63+
else if (map[current.row][current.col] == 'v') {
64+
next.row = current.row + 1;
65+
next.col = current.col;
66+
}
67+
else if (map[current.row][current.col] == '<') {
68+
next.row = current.row;
69+
next.col = current.col - 1;
70+
}
71+
else {
72+
std::cout << "this should not happen" << '\n';
73+
}
74+
if(in_limits(map, next.row, next.col) && map[next.row][next.col] != '#' && !seen[next.row][next.col]) {
75+
stack.push(next);
76+
dfs(stack, map, steps + 1, max_steps, dest, seen);
77+
stack.pop();
78+
}
79+
} else {
80+
for (const auto& motion : motions) {
81+
const auto next = current + motion;
82+
if(in_limits(map, next.row, next.col) && map[next.row][next.col] != '#' && !seen[next.row][next.col]) {
83+
stack.push(next);
84+
dfs(stack, map, steps + 1, max_steps, dest, seen);
85+
stack.pop();
86+
}
87+
}
88+
}
89+
seen[current.row][current.col] = false;
90+
}
91+
}
92+
93+
int main(int argc, char * argv[]) {
94+
std::string input = "../input/day_24_input";
95+
if (argc > 1) {
96+
input = argv[1];
97+
}
98+
99+
std::string line;
100+
std::fstream file(input);
101+
std::vector<std::string> map;
102+
while(std::getline(file, line)) {
103+
map.emplace_back(line);
104+
}
105+
106+
// std::priority_queue<Point, std::vector<Point>, Comparator> pq;
107+
std::stack<Point> stack;
108+
std::vector<std::vector<bool>> seen(map.size(), std::vector<bool>(map[0].size(), false));
109+
110+
Point current;
111+
current.row = 0;
112+
for (int col = 0; col < map[0].size(); col++) {
113+
if (map[0][col] == '.') {
114+
current.col = col;
115+
break;
116+
}
117+
}
118+
119+
Point dest;
120+
dest.row = map.size() - 1;
121+
for (int col = 0; col < map[dest.row].size(); col++) {
122+
if (map[dest.row][col] == '.') {
123+
dest.col = col;
124+
break;
125+
}
126+
}
127+
128+
stack.push(current);
129+
int max_steps = 0;
130+
dfs(stack, map, 0, max_steps, dest, seen);
131+
std::cout << max_steps << '\n';
132+
return 0;
133+
}

‎2023/cpp/day_23b.cpp

+184
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,184 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <queue>
7+
#include <stack>
8+
#include <string>
9+
#include <unordered_set>
10+
#include <vector>
11+
12+
// IDEA:
13+
// Convert the maze to a graph where
14+
// nodes in the graph are points with more than 2 adjacent points that are not forests
15+
// edges in the graph connect 2 nodes (no nodes in the middle of an edge)
16+
// Run dfs on the graph
17+
18+
struct Point {
19+
int row;
20+
int col;
21+
std::size_t dist = 0;
22+
23+
Point(const int row = 0, const int col = 0) : row(row), col(col) {}
24+
25+
Point operator + (const Point& p) const {
26+
return Point (p.row + row, p.col + col);
27+
}
28+
29+
bool operator == (const Point& p) const {
30+
return p.row == row && p.col == col;
31+
}
32+
};
33+
34+
struct Edge {
35+
Point start;
36+
Point end;
37+
std::size_t dist;
38+
};
39+
40+
struct hasher_point {
41+
std::size_t operator() (const Point& p) const {
42+
return p.row + p.col;
43+
}
44+
};
45+
46+
47+
const std::vector<Point> motions ={
48+
Point(-1,0),
49+
Point(0,1),
50+
Point(1,0),
51+
Point(0, -1),
52+
};
53+
54+
bool in_limits(const std::vector<std::string>& map, const int row, const int col) {
55+
return row >= 0 && row < map.size() && col >= 0 && col < map[0].size();
56+
}
57+
58+
void dfs(std::stack<Point>& stack,
59+
std::unordered_map<Point, std::vector<Edge>, hasher_point>& node_to_edges_map,
60+
std::size_t& max_dist,
61+
const Point& dest,
62+
std::vector<std::vector<bool>>& seen) {
63+
if (!stack.empty()) {
64+
const Point current = stack.top();
65+
seen[current.row][current.col] = true;
66+
if (current == dest) {
67+
max_dist = std::max(current.dist, max_dist);
68+
seen[current.row][current.col] = false;
69+
} else{
70+
for (const auto& edge : node_to_edges_map[current]) {
71+
auto next = edge.end;
72+
next.dist = current.dist + edge.dist;
73+
if(!seen[next.row][next.col]) {
74+
stack.push(next);
75+
dfs(stack, node_to_edges_map, max_dist, dest, seen);
76+
stack.pop();
77+
}
78+
}
79+
}
80+
seen[current.row][current.col] = false;
81+
}
82+
}
83+
84+
int main(int argc, char * argv[]) {
85+
std::string input = "../input/day_23_input";
86+
if (argc > 1) {
87+
input = argv[1];
88+
}
89+
90+
std::string line;
91+
std::fstream file(input);
92+
std::vector<std::string> map;
93+
while(std::getline(file, line)) {
94+
map.emplace_back(line);
95+
}
96+
97+
Point start;
98+
start.row = 0;
99+
for (int col = 0; col < map[0].size(); col++) {
100+
if (map[0][col] == '.') {
101+
start.col = col;
102+
break;
103+
}
104+
}
105+
106+
Point dest;
107+
dest.row = map.size() - 1;
108+
for (int col = 0; col < map[dest.row].size(); col++) {
109+
if (map[dest.row][col] == '.') {
110+
dest.col = col;
111+
break;
112+
}
113+
}
114+
115+
// IDEA:
116+
// Convert the maze to a graph where
117+
// nodes in the graph are points with more than 2 adjacent points that are not forests
118+
// edges in the graph connect 2 nodes (no nodes in the middle of an edge)
119+
// Run dfs on the graph
120+
121+
// Find nodes
122+
std::stack<Point> stack;
123+
std::vector<std::vector<bool>> seen(map.size(), std::vector<bool>(map[0].size(), false));
124+
std::unordered_set<Point, hasher_point> nodes{start, dest}; // Assumes start and end have only 1 open adjacent point
125+
stack.push(start);
126+
while (!stack.empty()) {
127+
const Point current = stack.top();
128+
stack.pop();
129+
seen[current.row][current.col] = true;
130+
int count = 0;
131+
for (const auto& motion : motions) {
132+
const auto next = current + motion;
133+
if(in_limits(map, next.row, next.col) && map[next.row][next.col] != '#') {
134+
// std::cout << "Considering: " << next.row << ',' << next.col << '\n';
135+
count++;
136+
if(!seen[next.row][next.col]) stack.push(next);
137+
}
138+
}
139+
if (count > 2) {
140+
nodes.insert(current);
141+
// std::cout << "NODE: " << current.row << ',' << current.col << '\n';
142+
}
143+
}
144+
145+
// Find the length of each edge in the graph where an edge connects 2 nodes
146+
// There are no nodes in the middle of an edge
147+
std::unordered_map<Point, std::vector<Edge>, hasher_point> node_to_edges_map;
148+
for (const auto& node : nodes) {
149+
seen = std::vector<std::vector<bool>> (map.size(), std::vector<bool>(map[0].size(), false));
150+
while(!stack.empty()) stack.pop();
151+
stack.push(node);
152+
while (!stack.empty()) {
153+
const Point current = stack.top();
154+
stack.pop();
155+
seen[current.row][current.col] = true;
156+
for (const auto& motion : motions) {
157+
auto next = current + motion;
158+
next.dist = current.dist + 1;
159+
if(in_limits(map, next.row, next.col) && map[next.row][next.col] != '#' && !seen[next.row][next.col]) {
160+
if (nodes.find(next) == nodes.end()) {
161+
stack.push(next);
162+
} else{
163+
Edge e;
164+
e.start = node;
165+
e.end = next;
166+
e.dist = next.dist;
167+
node_to_edges_map[node].push_back(e);
168+
e.start = next;
169+
e.end = node;
170+
node_to_edges_map[node].push_back(e);
171+
}
172+
}
173+
}
174+
}
175+
}
176+
177+
// Run DFS on the nodes using the precalculated edge distances
178+
seen = std::vector<std::vector<bool>> (map.size(), std::vector<bool>(map[0].size(), false));
179+
stack.push(start);
180+
std::size_t max_dist = 0;
181+
dfs(stack, node_to_edges_map, max_dist, dest, seen);
182+
std::cout << max_dist << '\n';
183+
return 0;
184+
}

‎2023/cpp/day_24a.cpp

+114
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <limits>
7+
#include <queue>
8+
#include <regex>
9+
#include <stack>
10+
#include <string>
11+
#include <unordered_set>
12+
#include <vector>
13+
#include <cassert>
14+
15+
// constexpr double limit_min = 7;
16+
// constexpr double limit_max = 27;
17+
18+
constexpr double limit_min = 200000000000000;
19+
constexpr double limit_max = 400000000000000;
20+
21+
struct Line {
22+
// ax + by = c
23+
double a;
24+
double b;
25+
double c;
26+
};
27+
28+
struct Hailstone {
29+
std::array<double, 3> position;
30+
std::array<double, 3> velocity;
31+
};
32+
33+
Hailstone parse_input(const std::string& line) {
34+
const std::regex pattern(R"((-?[0-9]+),(-?[0-9]+),(-?[0-9]+)@(-?[0-9]+),(-?[0-9]+),(-?[0-9]+))");
35+
std::smatch match;
36+
std::regex_search(line, match, pattern);
37+
// std::cout << '|' << line << '|' << '\n';
38+
// std::cout << match[0] << '\n';
39+
Hailstone hs;
40+
hs.position[0] = std::stod(match[1]);
41+
hs.position[1] = std::stod(match[2]);
42+
hs.position[2] = std::stod(match[3]);
43+
hs.velocity[0] = std::stod(match[4]);
44+
hs.velocity[1] = std::stod(match[5]);
45+
hs.velocity[2] = std::stod(match[6]);
46+
return hs;
47+
}
48+
49+
Line calculate_trajectory(const Hailstone& hs) {
50+
Line line;
51+
if (hs.velocity[0] == 0) {
52+
line.a = 0;
53+
line.b = 1;
54+
line.c = hs.position[0];
55+
} else {
56+
line.a = -hs.velocity[1];
57+
line.b = hs.velocity[0];
58+
line.c = line.b * hs.position[1] + line.a * hs.position[0];
59+
// line.is_vertical = false;
60+
// std::cout << -line.a/line.b << ' ' << hs.velocity[1]/hs.velocity[0] << '\n';
61+
assert(-line.a/line.b == hs.velocity[1]/hs.velocity[0]);
62+
// std::cout << line.a << " x + " << line.b << " y = " << line.c << '\n';
63+
64+
// std::cout << hs.position[1] << ' ' << line.m * hs.position[0] + line.c << ' ' << std::fabs(hs.position[1] - line.m * hs.position[0] - line.c) << '\n';
65+
// assert(std::fabs(hs.position[1] - line.m * hs.position[0] - line.c) < 0.00001);
66+
}
67+
return line;
68+
}
69+
70+
int main(int argc, char * argv[]) {
71+
std::string input = "../input/day_24_input";
72+
if (argc > 1) {
73+
input = argv[1];
74+
}
75+
76+
std::string line;
77+
std::fstream file(input);
78+
std::vector<Hailstone> hailstones;
79+
std::vector<Line> trajectories;
80+
while(std::getline(file, line)) {
81+
line.erase(std::remove_if(line.begin(), line.end(), [](const char c) {return c == ' ';}), line.end());
82+
hailstones.emplace_back(parse_input(line));
83+
trajectories.emplace_back(calculate_trajectory(hailstones.back()));
84+
}
85+
86+
int n_intersections = 0;
87+
for (int i = 0; i < trajectories.size(); i++) {
88+
const auto& line_1 = trajectories[i];
89+
for (int j = i+1; j < trajectories.size(); j++) {
90+
const auto& line_2 = trajectories[j];
91+
const auto determinant = (line_1.a * line_2.b) - (line_2.a * line_1.b);
92+
// std::cout << "\nStart: " << '(' << i << "," << j << ')' << '\n';
93+
// std::cout << determinant << '\n';
94+
if (determinant == 0) continue;
95+
const auto x = ((line_1.c * line_2.b) - (line_2.c * line_1.b)) / determinant;
96+
const auto y = ((line_1.a * line_2.c) - (line_2.a * line_1.c)) / determinant;
97+
if ((x - hailstones[i].position[0])/hailstones[i].velocity[0] < 0 ) {
98+
// std::cout << "intersected in the past" << '\n';
99+
continue;
100+
}
101+
if ((x - hailstones[j].position[0])/hailstones[j].velocity[0] < 0 ) {
102+
// std::cout << "intersected in the past" << '\n';
103+
continue;
104+
}
105+
// std::cout << x << ' ' << y << '\n';
106+
if (x >= limit_min && x <= limit_max && y >= limit_min && y <= limit_max) {
107+
// std::cout << "Incrementing" << '\n';
108+
n_intersections++;
109+
}
110+
}
111+
}
112+
std::cout << n_intersections << '\n';
113+
return 0;
114+
}

‎2023/cpp/day_24b.cpp

+256
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,256 @@
1+
#include <algorithm>
2+
#include <cmath>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <numeric>
6+
#include <limits>
7+
#include <queue>
8+
#include <regex>
9+
#include <stack>
10+
#include <string>
11+
#include <unordered_set>
12+
#include <vector>
13+
#include <cassert>
14+
15+
// Let (x0, y0, z0) and (vx0, vy0, vz0) be the position from and velocity with which the rock must be thrown
16+
// Let (xi, yi, zi) and (vxi, vyi, vzi) be the position and velocity of the ith hailstone, where i ranges from 1 to some `n`
17+
// hence there exists a time ti for every hailstone where the hailstone and the rock are at exactly the same position
18+
// hence
19+
// x0 + vx0 * ti = xi + vxi * ti ... (1)
20+
// => (x0 - xi) = - (vx0 - vxi) * ti
21+
// => (x0 - xi) / (vx0 - vxi) = -ti ... (2) (Assuming vx0 != vxi)
22+
// Similarly
23+
// (y0 - yi) / (vy0 - vyi) = - ti ... (3)
24+
// (z0 - zi) / (vz0 - vzi) = - ti ... (4)
25+
// Since the LHSs of (2) and (3) are both ti
26+
// (x0 - xi) / (vx0 - vxi) = (y0 - yi) / (vy0 - vyi)
27+
// (x0 - xi) * (vy0 - vyi) = (y0 - yi) * (vx0 - vxi)
28+
// x0 * vy0 - xi * vy0 - x0 * vyi + xi * vyi = y0 * vx0 - yi * vx0 - y0 * vxi + yi * vxi
29+
// This hold true for any i
30+
// Let i = 1: x0 * vy0 - x1 * vy0 - x0 * vy1 + x1 * vy1 = y0 * vx0 - y1 * vx0 - y0 * vx1 + y1 * vx1 ... (5)
31+
// Let i = 2: x0 * vy0 - x2 * vy0 - x0 * vy2 + x2 * vy2 = y0 * vx0 - y2 * vx0 - y0 * vx2 + y2 * vx2 ... (6)
32+
// Subtract (6) from (5):
33+
// (-x1 + x2) * vy0 + (-vy1 + vy2) * x0 + (x1 * vy1) - (x2 * vy2)
34+
// = (-y1 + y2) * vx0 + (-vx1 + vx2) * y0 +( y1 * vx1) + (y2 * vx2)
35+
// Rearranging the terms:
36+
// (-vy1 + vy2) * x0 - (-vx1 + vx2) * y0 - (-y1 + y2) * vx0 + (-x1 + x2) * vy0
37+
// = -(x1 * vy1) + (x2 * vy2) +( y1 * vx1) + (y2 * vx2)
38+
// Where all the coefficients of the position and velocities of the hailstones 1 and 2 are known, so this becomes
39+
// c1 * x0 + c2 * y0 + c3 * vx0 + c4 * vy0 = c5 where ci is some known constant ... (7)
40+
// Also, while (7) was obtained using ((1) and (2)), observe that this sort of an equation can be reached even by using ((2) and (4)) or ((3) and (4))
41+
// c6 * y0 + c7 * z0 + c8 * vy0 + c9 * vz0 = c10 where ci is some known constant
42+
// c11 * y0 + c12 * z0 + c13 * vy0 + c14 * vz0 = c15 where ci is some known constant
43+
// Also note that while here hailstones 1 and 2 were chosen, this can be done for any pair of hailstones
44+
// In (7) there are 4 unknowns, so by taking 4 pairs of hailstones, 4 equations can be obtained
45+
// These can then be put in a matrix:
46+
// [Some ][x0 ] = [Some other]
47+
// [matrix ][y0 ] [matrix ]
48+
// [of ][vx0] [of ]
49+
// [constants][vy0] [constants ]
50+
// Multiplying both sides by the inverse of first matrix gives
51+
// [x0 ] = Inverse([Some ]) * [Some other]
52+
// [y0 ] = [matrix ] [matrix ]
53+
// [vx0] = [of ] [of ]
54+
// [vy0] = [constants] [constants ]
55+
// This provides 4 out of th e6 unknowns
56+
// For the other 2 the same process is repeated, but with 2 equations this time with y and z instead of x and y
57+
// Only 2 equations are needed since the values of y have already been calculated,
58+
// but the values of y can be recalculated using 4 equations to sanity check the solution
59+
// This provides all the unknowns and hence the answer
60+
61+
struct Hailstone {
62+
std::array<long double, 3> position;
63+
std::array<long double, 3> velocity;
64+
};
65+
66+
Hailstone parse_input(const std::string& line) {
67+
const std::regex pattern(R"((-?[0-9]+),(-?[0-9]+),(-?[0-9]+)@(-?[0-9]+),(-?[0-9]+),(-?[0-9]+))");
68+
std::smatch match;
69+
std::regex_search(line, match, pattern);
70+
Hailstone hs;
71+
hs.position[0] = std::stod(match[1]);
72+
hs.position[1] = std::stod(match[2]);
73+
hs.position[2] = std::stod(match[3]);
74+
hs.velocity[0] = std::stod(match[4]);
75+
hs.velocity[1] = std::stod(match[5]);
76+
hs.velocity[2] = std::stod(match[6]);
77+
return hs;
78+
}
79+
80+
template<size_t N>
81+
constexpr std::array<std::array<long double, N>, N> inverse_using_LU_decomp(std::array<std::array<long double, N>, N> mat) {
82+
std::array<std::array<long double, N>, N> p;
83+
std::array<std::array<long double, N>, N> inversed_data;
84+
for (int i = 0; i < N; i++) {
85+
for (int j = 0; j < N; j++) {
86+
if (i == j) p[i][j] = 1;
87+
else p[i][j] = 0;
88+
}
89+
}
90+
for (int i = 0; i < N - 1; i++){
91+
auto max_idx = i;
92+
for (int k = i+1; k < N; k++) {
93+
if (std::abs(mat[k][i]) > std::abs(mat[max_idx][i])) {
94+
max_idx = k;
95+
}
96+
}
97+
98+
std::swap(mat[i], mat[max_idx]);
99+
std::swap(p[i], p[max_idx]);
100+
101+
for (int j = i+1; j < N; j++) {
102+
mat[j][i] /= mat[i][i];
103+
for (int k = i+1; k < N; k++) {
104+
mat[j][k] -= mat[j][i] * mat[i][k];
105+
}
106+
}
107+
}
108+
109+
for (int i_main = 0; i_main < N; i_main++) {
110+
std::array<long double, N> b;
111+
for (int j = 0; j < N; j++) {
112+
if (i_main == j) {
113+
b[j] = 1;
114+
} else {
115+
b[j] = 0;
116+
}
117+
}
118+
119+
auto temp = b;
120+
for (int i = 0; i < N; i++) {
121+
temp[i] = std::inner_product(std::begin(p[i]), std::end(p[i]), std::begin(b), 0.);
122+
}
123+
b = temp;
124+
125+
for (int i = 0; i < N-1; i++){
126+
for (int j = i+1; j < N; j++) {
127+
b[j] -= mat[j][i] * b[i];
128+
}
129+
}
130+
131+
for (int i = N-1; i >=0; i--) {
132+
b[i] /= mat[i][i];
133+
for (int j = 0; j < i; j++) {
134+
b[j] -= mat[j][i] * b[i];
135+
}
136+
}
137+
138+
for (int k =0; k < N; k++) {
139+
inversed_data[k][i_main] = b[k];
140+
}
141+
}
142+
return inversed_data;
143+
}
144+
145+
// The function get_equation below provides the cofficients of equation 7
146+
// With the returned array containing c1, c2, c3, c4 and the returned double containing c5
147+
// idx0 and idx1 help choose whether the function uses the coefficients of
148+
// (x,y) or (y,z) or (x,z) by using the values
149+
// (0,1) or (1,2) or (0,2)
150+
std::tuple<bool, std::array<long double, 4>, long double> get_equation(const Hailstone& hs1, const Hailstone& hs2, const int idx0, const int idx1) {
151+
if (hs1.position[idx1] == hs2.position[idx1]) return {false, std::array<long double, 4>(), 0};
152+
if (hs1.velocity[idx1] == hs2.velocity[idx1]) return {false, std::array<long double, 4>(), 0};
153+
const std::tuple<bool, std::array<long double, 4>, long double> ans = {
154+
true,
155+
{
156+
(hs2.velocity[idx1] - hs1.velocity[idx1]), // coeffecient of x0 if idx0 = 0
157+
-(hs2.velocity[idx0] - hs1.velocity[idx0]), // coeffecient of y0 if idx1 = 1
158+
-(hs2.position[idx1] - hs1.position[idx1]), // coeffecient of vx0 if idx0 = 0
159+
(hs2.position[idx0] - hs1.position[idx0]) // coeffecient of vy0 if idx1 = 1
160+
},
161+
((hs1.position[idx1] * hs1.velocity[idx0]) - (hs1.position[idx0] * hs1.velocity[idx1]))
162+
- ((hs2.position[idx1] * hs2.velocity[idx0]) - (hs2.position[idx0] * hs2.velocity[idx1])) // RHS
163+
};
164+
// std::cout << std::get<1>(ans)[0] << ' '
165+
// << std::get<1>(ans)[1] << ' '
166+
// << std::get<1>(ans)[2] << ' '
167+
// << std::get<1>(ans)[3] << " | "
168+
// << std::get<2>(ans) << '\n';
169+
return ans;
170+
}
171+
172+
int main(int argc, char * argv[]) {
173+
std::string input = "../input/day_24_input";
174+
if (argc > 1) {
175+
input = argv[1];
176+
}
177+
178+
std::string line;
179+
std::fstream file(input);
180+
std::vector<Hailstone> hailstones;
181+
while(std::getline(file, line)) {
182+
line.erase(std::remove_if(line.begin(), line.end(), [](const char c) {return c == ' ';}), line.end());
183+
hailstones.emplace_back(parse_input(line));
184+
}
185+
std::array<long double, 3> ans_position;
186+
std::array<long double, 3> ans_velocity;
187+
// Calculating for (x,y)
188+
{
189+
int count = 0;
190+
std::array<std::array<long double, 4>, 4> main_lhs;
191+
std::array<long double, 4> main_rhs;
192+
for(int i = 0; i < hailstones.size()-1 && count < 4; i++) {
193+
for(int j = i+1; j < hailstones.size() && count < 4; j++) {
194+
const auto [valid, lhs, rhs] = get_equation(hailstones[i], hailstones[j], 0, 1);
195+
if (valid) {
196+
main_lhs[count] = lhs;
197+
main_rhs[count] = rhs;
198+
count++;
199+
}
200+
}
201+
}
202+
const auto inverse = inverse_using_LU_decomp<4>(main_lhs);
203+
std::array<long double, 4> answer_of_calcs {0,0,0,0};
204+
for (int row = 0; row < 4; row++) {
205+
for (int col = 0; col < 4; col++) {
206+
answer_of_calcs[row] += inverse[row][col] * main_rhs[col];
207+
}
208+
}
209+
ans_position[0] = answer_of_calcs[0];
210+
ans_position[1] = answer_of_calcs[1];
211+
ans_velocity[0] = answer_of_calcs[2];
212+
ans_velocity[1] = answer_of_calcs[3];
213+
}
214+
// Calculating for (x,z)
215+
{
216+
int count = 0;
217+
std::array<std::array<long double, 4>, 4> main_lhs;
218+
std::array<long double, 4> main_rhs;
219+
for(int i = 0; i < hailstones.size()-1 && count < 4; i++) {
220+
for(int j = i+1; j < hailstones.size() && count < 4; j++) {
221+
const auto [valid, lhs, rhs] = get_equation(hailstones[i], hailstones[j], 1, 2);
222+
if (valid) {
223+
main_lhs[count] = lhs;
224+
main_rhs[count] = rhs;
225+
count++;
226+
}
227+
}
228+
}
229+
const auto inverse = inverse_using_LU_decomp<4>(main_lhs);
230+
std::array<long double, 4> answer_of_calcs {0,0,0,0};
231+
for (int row = 0; row < 4; row++) {
232+
for (int col = 0; col < 4; col++) {
233+
answer_of_calcs[row] += inverse[row][col] * main_rhs[col];
234+
}
235+
}
236+
// assert(ans_position[0] == answer_of_calcs[0]);
237+
// assert(ans_position[2] == answer_of_calcs[2]);
238+
ans_position[2] = answer_of_calcs[1];
239+
ans_velocity[2] = answer_of_calcs[3];
240+
}
241+
// std::cout << "Answers: " << '\n';
242+
// for (const auto& ele : ans_position) {
243+
// std::cout << std::fixed << ele << ' ';
244+
// }
245+
// std::cout << '\n';
246+
// for (const auto& ele : ans_velocity) {
247+
// std::cout << std::fixed << ele << ' ';
248+
// }
249+
// std::cout << '\n';
250+
long long answer = 0;
251+
for (const auto& ele : ans_position) {
252+
answer += static_cast<long long>(ele);
253+
}
254+
std::cout << std::fixed << answer << '\n';
255+
return 0;
256+
}

‎2023/cpp/day_25a.cpp

+186
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,186 @@
1+
#include <algorithm>
2+
#include <numeric>
3+
#include <fstream>
4+
#include <iostream>
5+
#include <string>
6+
#include <string_view>
7+
#include <vector>
8+
#include <unordered_set>
9+
#include <unordered_map>
10+
#include <random>
11+
#include <cassert>
12+
#include <stack>
13+
14+
// Used Karger's algorithm
15+
// Refactor this; should run much faster
16+
// Also, other algos are available
17+
18+
struct Edge {
19+
std::string start;
20+
std::string end;
21+
std::size_t hash = 0;
22+
bool operator == (const Edge& e) const{
23+
return start == e.start && end == e.end;
24+
}
25+
void update_hash() {
26+
std::size_t p = 1;
27+
for (int i = 0; i < start.size(); i++) {
28+
hash += p * (start[i] - 'a');
29+
p*= 10;
30+
}
31+
for (int i = 0; i < end.size(); i++) {
32+
hash += p * (start[i] - 'a');
33+
p *= 10;
34+
}
35+
}
36+
};
37+
38+
struct hasher {
39+
std::size_t operator () (const Edge& e) const {
40+
return e.hash;
41+
42+
}
43+
};
44+
45+
void parse_input(const std::string& input_str, std::unordered_set<Edge, hasher>& edges) {
46+
const std::string from = input_str.substr(0, 3);
47+
std::size_t start = 5;
48+
std::size_t end = input_str.find(' ', start);
49+
while (end != std::string::npos) {
50+
const std::string to = input_str.substr(start, end - start);
51+
start = end + 1;
52+
end = input_str.find(' ', start);
53+
Edge edge;
54+
edge.start = from;
55+
edge.end = to;
56+
edge.update_hash();
57+
edges.insert(edge);
58+
}
59+
const std::string to = input_str.substr(start, input_str.size() - start);
60+
Edge edge;
61+
edge.start = from;
62+
edge.end = to;
63+
edges.insert(edge);
64+
}
65+
66+
const std::string& UnionFindRCFindUtil(const std::string& v, std::unordered_map<std::string, std::pair<std::string, int>>& subsets){
67+
if(v != subsets[v].first) {
68+
subsets[v].first = UnionFindRCFindUtil(subsets[v].first, subsets);
69+
}
70+
return subsets[v].first;
71+
}
72+
73+
void UnionFindRCUnionUtil(const std::string& v1, const std::string& v2, std::unordered_map<std::string, std::pair<std::string, int>>& subsets) {
74+
const auto& p1 = UnionFindRCFindUtil(v1, subsets);
75+
const auto& p2 = UnionFindRCFindUtil(v2, subsets);
76+
77+
if(subsets[p1].second > subsets[p2].second) {
78+
subsets[p2].first = p1;
79+
} else if(subsets[p1].second < subsets[p1].second) {
80+
subsets[p1].first = p2;
81+
} else {
82+
subsets[p2].first = p1;
83+
subsets[p1].second++;
84+
}
85+
}
86+
87+
std::random_device dev;
88+
std::mt19937 rng(dev());
89+
std::uniform_int_distribution<std::mt19937::result_type> dist(1000);
90+
91+
void KargersAlgorithm(std::vector<Edge> edges,
92+
std::size_t n_vertices,
93+
std::unordered_map<std::string, std::pair<std::string, int>> subsets,
94+
std::unordered_map<Edge, int, hasher>& answers) {
95+
int count = edges.size();
96+
while (n_vertices > 2) {
97+
const auto idx = dist(rng) % (count);
98+
const auto& edge = edges[idx];
99+
const auto& p1 = UnionFindRCFindUtil(edge.start, subsets);
100+
const auto& p2 = UnionFindRCFindUtil(edge.end, subsets);
101+
if (p1 != p2) {
102+
UnionFindRCUnionUtil(p1, p2, subsets);
103+
n_vertices--;
104+
}
105+
count--;
106+
std::swap(edges[idx], edges[count]);
107+
}
108+
for (int i = 0; i < count; i++) {
109+
const auto& p1 = UnionFindRCFindUtil(edges[i].start, subsets);
110+
const auto& p2 = UnionFindRCFindUtil(edges[i].end, subsets);
111+
if (p1 != p2) {
112+
answers[edges[i]]++;
113+
}
114+
}
115+
}
116+
117+
int main(int argc, char * argv[]) {
118+
std::string input = "../input/day_25_input";
119+
if (argc > 1) {
120+
input = argv[1];
121+
}
122+
123+
std::string line;
124+
std::fstream file(input);
125+
std::unordered_set<Edge, hasher> edges_main;
126+
while(std::getline(file, line)) {
127+
parse_input(line, edges_main);
128+
}
129+
130+
dist = std::uniform_int_distribution<std::mt19937::result_type>(edges_main.size());
131+
std::unordered_set<std::string> vertices;
132+
std::vector<Edge> edges_v;
133+
for (const auto& edge : edges_main) {
134+
vertices.insert(edge.start);
135+
vertices.insert(edge.end);
136+
edges_v.push_back(edge);
137+
}
138+
139+
std::unordered_map<Edge, int, hasher> answers;
140+
141+
std::unordered_map<std::string, std::pair<std::string, int>> subsets;
142+
for (const auto& v : vertices) {
143+
subsets[v] = {v, 0};
144+
}
145+
const auto n_vertices = vertices.size();
146+
for (int iteration = 0; iteration < 10000; iteration++) {
147+
if (iteration % 100 == 0) std::cout << "Iteration: " << iteration << '\n';
148+
KargersAlgorithm(edges_v, n_vertices, subsets, answers);
149+
}
150+
// Create the edge list with the top 3 edges removed.
151+
auto edges = edges_main;
152+
{
153+
std::vector<std::pair<Edge, int>> answers_v;
154+
for (const auto& [edge, count] : answers) {
155+
answers_v.push_back({edge, count});
156+
}
157+
std::partial_sort(std::begin(answers_v), std::begin(answers_v) + 3, std::end(answers_v), [](const auto& e1, const auto& e2) { return e1.second > e2.second; });
158+
edges.erase(answers_v[0].first);
159+
edges.erase(answers_v[1].first);
160+
edges.erase(answers_v[2].first);
161+
}
162+
// Find all the points connecting to a single (random) point to find one of the disjoint sets
163+
std::unordered_map<std::string, std::unordered_set<std::string>> adj_list;
164+
for (const auto& e : edges) {
165+
adj_list[e.start].insert(e.end);
166+
adj_list[e.end].insert(e.start);
167+
}
168+
std::stack<std::string> stack;
169+
stack.push(adj_list.begin()->first);
170+
std::unordered_set<std::string> seen;
171+
while (!stack.empty()) {
172+
const auto current = stack.top();
173+
stack.pop();
174+
if (seen.find(current) != seen.end()) {
175+
continue;
176+
}
177+
seen.insert(current);
178+
for (const auto& ele : adj_list[current]) {
179+
if (seen.find(ele) == seen.end()) {
180+
stack.push(ele);
181+
}
182+
}
183+
}
184+
std::cout << seen.size() * (vertices.size() - seen.size()) << '\n';
185+
return 0;
186+
}

‎2023/sample_input/day_20_sample_input

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
broadcaster -> a, b, c
2+
%a -> b
3+
%b -> c
4+
%c -> inv
5+
&inv -> a

‎2023/sample_input/day_21_sample_input

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
...........
2+
.....###.#.
3+
.###.##..#.
4+
..#.#...#..
5+
....#.#....
6+
.##..S####.
7+
.##..#...#.
8+
.......##..
9+
.##.#.####.
10+
.##..##.##.
11+
...........

‎2023/sample_input/day_22_sample_input

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
1,0,1~1,2,1
2+
0,0,2~2,0,2
3+
0,2,3~2,2,3
4+
0,0,4~0,2,4
5+
2,0,5~2,2,5
6+
0,1,6~2,1,6
7+
1,1,8~1,1,9

‎2023/sample_input/day_23_sample_input

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
#.#####################
2+
#.......#########...###
3+
#######.#########.#.###
4+
###.....#.>.>.###.#.###
5+
###v#####.#v#.###.#.###
6+
###.>...#.#.#.....#...#
7+
###v###.#.#.#########.#
8+
###...#.#.#.......#...#
9+
#####.#.#.#######.#.###
10+
#.....#.#.#.......#...#
11+
#.#####.#.#.#########v#
12+
#.#...#...#...###...>.#
13+
#.#.#v#######v###.###v#
14+
#...#.>.#...>.>.#.###.#
15+
#####v#.#.###v#.#.###.#
16+
#.....#...#...#.#.#...#
17+
#.#########.###.#.#.###
18+
#...###...#...#...#.###
19+
###.###.#.###v#####v###
20+
#...#...#.#.>.>.#.>.###
21+
#.###.###.#.###.#.#v###
22+
#.....###...###...#...#
23+
#####################.#

‎2023/sample_input/day_24_sample_input

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
19, 13, 30 @ -2, 1, -2
2+
18, 19, 22 @ -1, -1, -2
3+
20, 25, 34 @ -2, -2, -4
4+
12, 31, 28 @ -1, -2, -1
5+
20, 19, 15 @ 1, -5, -3

‎2023/sample_input/day_25_sample_input

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
jqt: rhn xhk nvd
2+
rsh: frs pzl lsr
3+
xhk: hfx
4+
cmg: qnr nvd lhk bvb
5+
rhn: xhk bvb hfx
6+
bvb: xhk hfx
7+
pzl: lsr hfx nvd
8+
qnr: nvd
9+
ntq: jqt hfx bvb xhk
10+
nvd: lhk
11+
lsr: lhk
12+
rzs: qnr cmg lsr rsh
13+
frs: qnr lhk lsr

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ This repository contains solutions to the Advent of Code puzzles.
66

77
| Year | C++ | Python | Link to folders | Link to README.md |
88
|:----:|:----:|:----:|:----:|:----:|
9-
|2023 |1-19 | - | [Link](/2023/) |[Link](/2023/README.md) |
9+
|2023 |1-25 | - | [Link](/2023/) |[Link](/2023/README.md) |
1010
|2022 |1-25 | - | [Link](/2022/) |[Link](/2022/README.md) |
1111
|2021 |1-25 | - | [Link](/2021/) |[Link](/2021/README.md) |
1212
|2020 |1-25 |1-25 | [Link](/2020/) |[Link](/2020/README.md) |

0 commit comments

Comments
 (0)
Please sign in to comment.