diff --git a/README.md b/README.md index f5d628d..600a258 100644 --- a/README.md +++ b/README.md @@ -90,6 +90,14 @@ Adding up the products of area and perimeter, or area and corner count, per regi For each claw machine, we have to solve a simple linear equation system to get the number of required button pushes. Conveniently, for the given data, all of these systems have a unique solution. However, some solutions have to be excluded because the number of button pushes is not integer, or because the solutions are too high for part 1. The required integer tests are done by performing integer division, and comparing the dividend with the product of the integer quotient and the divisor. +### Day 14: Restroom Redoubt + +:mag_right: Puzzle: , :white_check_mark: Solver: [`RestroomRedoubt.cpp`](src/RestroomRedoubt.cpp) + +For part 1, the resulting quadrant for each robot can be calculated directly, by multiplying the velocity vector by 100 and adding it to the position vector, modulo width or height to stay within the target range. + +When observing a few resulting robot configurations after each passed second, it becomes apparent that some of them show vertical or horizontal bands of clustering, and that these repeat every `width` or `height` seconds, respectively. With a bit of modulo calculus, the time after which the vertical and horizontal bands overlap can be determined, which is the solution for part 2. + ## Thanks * [Alexander Brouwer](https://github.com/Bromvlieg) for getting the project set up with CMake. diff --git a/include/aoc/Math.hpp b/include/aoc/Math.hpp index ab27ac3..a0a125d 100644 --- a/include/aoc/Math.hpp +++ b/include/aoc/Math.hpp @@ -15,8 +15,26 @@ #pragma once +#include + class Math { public: + /// + /// Calculates an integer exponentiation. + /// + /// Base of the exponentiation. + /// Exponent of the exponentiation + /// 'base' raised to the power of 'exponent'. static int ipow(const int base, const int exponent); + + /// + /// Calculates the greatest common divisor gcd(a, b) and the coefficients x and y of Bézout's identity + /// ax + by = gcd(a, b). If a and b are coprime, then x is the modular multiplicative inverse of a modulo b, and y + /// is the modular multiplicative inverse of b modulo a. + /// + /// + /// + /// A tuple of the gcd(a, b), x, and y. + static std::tuple extendedEuclid(const int a, const int b); }; diff --git a/include/aoc/Point2.hpp b/include/aoc/Point2.hpp index 857471f..82cc65d 100644 --- a/include/aoc/Point2.hpp +++ b/include/aoc/Point2.hpp @@ -43,4 +43,5 @@ class Point2 Point2& operator+=(const Point2& rhs); Point2& operator-=(const Point2& rhs); Point2& operator*=(const int rhs); + int& operator[](size_t coordinateIndex); }; diff --git a/include/aoc/RestroomRedoubt.hpp b/include/aoc/RestroomRedoubt.hpp index 65e37d3..0b93a6e 100644 --- a/include/aoc/RestroomRedoubt.hpp +++ b/include/aoc/RestroomRedoubt.hpp @@ -16,14 +16,16 @@ #pragma once #include +#include +#include #include class RestroomRedoubt : public Solver { public: - RestroomRedoubt(const int width = 101, const int height = 103); + RestroomRedoubt(const int width = 101, const int height = 103, const bool runPart2 = true); virtual const std::string getPuzzleName() const override; virtual const int getPuzzleDay() const override; virtual void processDataLine(const std::string& line) override; @@ -33,6 +35,10 @@ class RestroomRedoubt const int height_; const int halfWidth_; const int halfHeight_; + const bool runPart2_; std::array quadrants_; + std::vector> robots_; static constexpr int getNPredictionSeconds(); + static constexpr int getEasterEggThreshold(); + void findEasterEgg(); }; diff --git a/src/Math.cpp b/src/Math.cpp index a3769b5..760ed28 100644 --- a/src/Math.cpp +++ b/src/Math.cpp @@ -15,6 +15,8 @@ #include +#include + int Math::ipow(const int base, const int exponent) { int result = 1; @@ -24,3 +26,22 @@ int Math::ipow(const int base, const int exponent) } return result; } + +std::tuple Math::extendedEuclid(const int a, const int b) +{ + std::array, 3> rst{ { { a, b }, { 1, 0 }, { 0, 1 } } }; + int q; + int r; + + while (rst[0].second > 0) + { + q = rst[0].first / rst[0].second; + for (auto& p : rst) + { + r = p.first - q * p.second; + p.first = p.second; + p.second = r; + } + } + return { rst[0].first, rst[1].first, rst[2].first }; +} diff --git a/src/Point2.cpp b/src/Point2.cpp index 6437d53..004289f 100644 --- a/src/Point2.cpp +++ b/src/Point2.cpp @@ -84,3 +84,8 @@ Point2& Point2::operator*=(const int rhs) *this = *this * rhs; return *this; } + +int& Point2::operator[](size_t coordinateIndex) +{ + return coordinateIndex == 0 ? x : y; +} diff --git a/src/RestroomRedoubt.cpp b/src/RestroomRedoubt.cpp index 23b6f3f..2ef2f09 100644 --- a/src/RestroomRedoubt.cpp +++ b/src/RestroomRedoubt.cpp @@ -15,14 +15,17 @@ #include +#include +#include #include #include #include +#include #include -RestroomRedoubt::RestroomRedoubt(const int width, const int height) - : width_{ width }, height_{ height }, halfWidth_{ width / 2 }, halfHeight_{ height / 2 } +RestroomRedoubt::RestroomRedoubt(const int width, const int height, const bool runPart2) + : width_{ width }, height_{ height }, halfWidth_{ width / 2 }, halfHeight_{ height / 2 }, runPart2_{ runPart2 } { quadrants_.fill(0); } @@ -41,10 +44,10 @@ void RestroomRedoubt::processDataLine(const std::string& line) { std::istringstream stream{ line }; char c; - Point2 p; - Point2 v; + Point2 p, v; stream >> c >> c >> p.x >> c >> p.y >> c >> c >> v.x >> c >> v.y; + // Calculates the resulting position, and shifts negative results back into the target range. Point2 position = p + v * getNPredictionSeconds(); position.x %= width_; if (position.x < 0) @@ -57,6 +60,7 @@ void RestroomRedoubt::processDataLine(const std::string& line) position.y += height_; } + // Determines the resulting quadrant from the position. if (position.x < halfWidth_) { if (position.y < halfHeight_) @@ -79,14 +83,72 @@ void RestroomRedoubt::processDataLine(const std::string& line) quadrants_[3]++; } } + + // Keeps track of the robot for part 2. + robots_.push_back({ p, v }); } void RestroomRedoubt::finish() { part1 = std::accumulate(quadrants_.cbegin(), quadrants_.cend(), 1, std::multiplies{}); + + if (runPart2_) + { + findEasterEgg(); + } } constexpr int RestroomRedoubt::getNPredictionSeconds() { return 100; } + +constexpr int RestroomRedoubt::getEasterEggThreshold() +{ + return 25; +} + +void RestroomRedoubt::findEasterEgg() +{ + std::array patterns{ 0, 0 }; + std::array sizes{ width_, height_ }; + for (size_t i = 0; i < 2; i++) + { + int seconds{ 0 }; + while (patterns[i] == 0) + { + std::vector buckets(sizes[i], 0); + for (auto& robot : robots_) + { + buckets[robot.first[i]]++; + robot.first[i] += robot.second[i]; + if (robot.first[i] < 0) + { + robot.first[i] += sizes[i]; + } + else if (robot.first[i] >= sizes[i]) + { + robot.first[i] -= sizes[i]; + } + } + auto it = std::find_if(buckets.cbegin(), buckets.cend(), [](int x) { return x > getEasterEggThreshold(); }); + if (it != buckets.cend()) + { + patterns[i] = seconds; + } + seconds++; + } + } + + // Determines the iteration where the repeating patterns of horizontal and vertical bands overlap, that is + // patterns_0 + width * x = patterns_1 + height * y + // for two positive integers 'x' and 'y'. This is equivalent to finding 'x' such that + // width * x ≡ patterns_1 - patterns_0 (modulo height) + // We can determine the modular multiplicative inverse of 'width' modulo 'height' with the extended Euclidean + // algorithm. + const auto [gcd, invHeight, invWidth] = Math::extendedEuclid(height_, width_); + // With that we can find 'x'. + int x{ (invWidth * (patterns[1] - patterns[0])) % height_ }; + // And calculate when the overlap occurs. + part2 = patterns[0] + width_ * x; +} diff --git a/tests/src/TestCases.cpp b/tests/src/TestCases.cpp index dd9b3f3..d333bb5 100644 --- a/tests/src/TestCases.cpp +++ b/tests/src/TestCases.cpp @@ -226,11 +226,11 @@ TEST_CASE("[RestroomRedoubtTests]") TestContext test; SECTION("FullData") { - test.run(std::make_unique(), 224969976, 0, test.getInputPaths()); + test.run(std::make_unique(), 224969976, 7892, test.getInputPaths()); } SECTION("ExampleData") { - test.run(std::make_unique(11, 7), 12, 0, test.getExampleInputPaths()); + test.run(std::make_unique(11, 7, false), 12, 0, test.getExampleInputPaths()); } }