summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFederico Igne <undyamon@disroot.org>2023-12-15 18:20:56 +0100
committerFederico Igne <undyamon@disroot.org>2023-12-15 18:20:56 +0100
commiteafa46252f2d41a86bf218b8af0460eeb5818eb9 (patch)
tree24a286059ba947b24b7b57c5a5e7c26980d029e8
parentfd2077171f5b0b1a9f3bb4224865f0d5c84c21ae (diff)
downloadaoc-eafa46252f2d41a86bf218b8af0460eeb5818eb9.tar.gz
aoc-eafa46252f2d41a86bf218b8af0460eeb5818eb9.zip
aoc(2315): Lens Library
-rw-r--r--2023/15/Makefile19
-rw-r--r--2023/15/resources/input_small.txt1
-rw-r--r--2023/15/src/part1.cpp33
-rw-r--r--2023/15/src/part2.cpp136
4 files changed, 189 insertions, 0 deletions
diff --git a/2023/15/Makefile b/2023/15/Makefile
new file mode 100644
index 0000000..af6a294
--- /dev/null
+++ b/2023/15/Makefile
@@ -0,0 +1,19 @@
1CXXFLAGS := -std=c++17
2CPPFLAGS := -I../include
3EXE := part1 part2
4
5.PHONY: all clean configure
6
7all: $(EXE)
8
9configure:
10 bear -- $(MAKE) all
11
12%.o: %.cpp
13 $(CXX) -c $(CPPFLAGS) $(CXXFLAGS) $< -o $@
14
15clean:
16 rm -rf $(EXE) src/*.o compile_commands.json
17
18%: src/%.o
19 $(CXX) $^ -o $@
diff --git a/2023/15/resources/input_small.txt b/2023/15/resources/input_small.txt
new file mode 100644
index 0000000..4f58f74
--- /dev/null
+++ b/2023/15/resources/input_small.txt
@@ -0,0 +1 @@
rn=1,cm-,qp=3,cm=2,qp-,pc=4,ot=9,ab=5,pc-,pc=6,ot=7
diff --git a/2023/15/src/part1.cpp b/2023/15/src/part1.cpp
new file mode 100644
index 0000000..6473d20
--- /dev/null
+++ b/2023/15/src/part1.cpp
@@ -0,0 +1,33 @@
1#include <iostream>
2#include <fstream>
3
4int main(int argc, char* argv[])
5{
6 int answer{};
7
8 if (std::ifstream input{ argv[1] }; input.is_open())
9 {
10 std::string line;
11 std::getline(input,line);
12
13 int hash{};
14 for (unsigned char c : line)
15 {
16 if (c == ',')
17 {
18 answer += hash;
19 hash = 0;
20 }
21 else
22 {
23 hash += c;
24 hash *= 17;
25 hash %= 256;
26 }
27 }
28 answer += hash;
29 }
30
31 std::cout << answer << std::endl;
32 return 0;
33}
diff --git a/2023/15/src/part2.cpp b/2023/15/src/part2.cpp
new file mode 100644
index 0000000..4758e86
--- /dev/null
+++ b/2023/15/src/part2.cpp
@@ -0,0 +1,136 @@
1#include <algorithm>
2#include <iostream>
3#include <memory>
4#include <fstream>
5#include <vector>
6#include <array>
7#include <tuple>
8
9#include "util.h"
10
11using Lens = std::pair<std::string,int>;
12using Box = std::vector<Lens>;
13using Boxes = std::array<Box,256>;
14
15struct Cmd
16{
17protected:
18
19 int boxi_;
20 std::string label_;
21
22public:
23
24 Cmd(const std::string& label) : boxi_{ hash(label) }, label_{label}
25 {
26 }
27
28 virtual void apply(Boxes& boxes) const
29 {
30 auto& box = boxes[boxi_];
31 auto has_same_label = [this](const auto& lens) { return lens.first == label_; };
32
33 if (auto lens = std::find_if(box.cbegin(), box.cend(), has_same_label);
34 lens != box.cend())
35 {
36 box.erase(lens);
37 }
38 }
39
40 static int hash(const std::string& label)
41 {
42 int hash{};
43 for (unsigned char c : label)
44 {
45 hash += c;
46 hash *= 17;
47 hash %= 256;
48 }
49 return hash;
50 }
51};
52
53class Add : public Cmd
54{
55 int focal_;
56
57public:
58
59 Add(const std::string& label, int focal) : Cmd(label), focal_{ focal }
60 {
61 }
62
63 void apply(Boxes& boxes) const override
64 {
65 auto& box = boxes[boxi_];
66 auto has_same_label = [this](const auto& lens) { return lens.first == label_; };
67
68 if (auto lens = std::find_if(box.begin(), box.end(), has_same_label);
69 lens != box.end())
70 {
71 lens->second = focal_;
72 }
73 else
74 {
75 box.push_back({ label_, focal_ });
76 }
77 }
78};
79
80int power(const Boxes& boxes)
81{
82 int power{};
83 for (int b = 0; b < boxes.size(); ++b)
84 {
85 for(int l = 0; l < boxes[b].size(); ++l)
86 {
87 power += (b + 1) * (l + 1) * boxes[b][l].second;
88 }
89 }
90 return power;
91}
92
93std::vector<std::unique_ptr<Cmd>> parse(const char* file)
94{
95 using namespace util::separators;
96
97 std::vector<std::unique_ptr<Cmd>> cmds;
98
99 if (std::ifstream input{ file }; input.is_open())
100 {
101 std::string line;
102 std::getline(input,line);
103 std::string_view view{ line };
104
105 auto to_cmd = [&cmds](std::string_view& str)
106 {
107 auto c{ str.find_first_of("=-") };
108 std::string label{ str.substr(0, c) };
109 if (str[c] == '-')
110 {
111 cmds.push_back(std::make_unique<Cmd>(label));
112 }
113 else
114 {
115 int focal = std::stoi( std::string{ str.substr(c + 1) } );
116 cmds.push_back(std::make_unique<Add>(label, focal));
117 }
118 };
119
120 util::accumulate<COMMA>(view, to_cmd);
121 }
122
123 return cmds;
124}
125
126int main(int argc, char* argv[])
127{
128 Boxes boxes;
129 for (const auto& cmd : parse(argv[1]))
130 {
131 cmd->apply(boxes);
132 }
133
134 std::cout << power(boxes) << std::endl;
135 return 0;
136}