Revisiting AoC 2022: The Journey to a 20,000x Speedup
Table of Contents
1. Advent of Code
Advent of Code (abbreviated AoC) is an annual Advent calendar of small programming puzzles. There’s a programming puzzle for each day in the calendar, accompanied by a story which ties the problems together.
After hearing about AoC from some friends in 2022, I decided to give it a go. About halfway through I came across a problem which I thought was particularly interesting: Day 15: Beacon Exclusion Zone. It’s now more than a year later, AoC 2023 has long concluded, and I’m finally getting around to publishing this… better late than never!
In this post I’m going to discuss 3 different approaches to solving this problem and compare their performance. I’ll introduce each approach in the order that I implemented them. I hope you’ll find this problem as interesting as I did.
2. The Problem
AoC problems come in two parts, with the second part building on the first. I want to focus on the aspects that I found most interesting, so I’m going to skip straight to the second part and gloss over some minor details. To get the full experience complete with Excuse Plot and all, you’ll need to look at the actual AoC problem description.
The general idea is that there are a bunch of “sensors” which can detect the area around them. The area “covered” by each sensor is dictated by the sensor’s position and range. The sensors are arranged in an area such that all but one point in the area is covered by at least one sensor. Our task is to find this uncovered point.
The AoC platform generates unique problem input data for each user, below is a visualisation of mine, it’ll make more sense as we go along:
Let’s dive into the details of the problem, we’ll start with a diagram, and I’ll attempt to explain how to interpret it from there:
The problem takes place on a 2D grid, I’ve placed gridlines at 1-unit intervals in the diagram to help visualise the scale and position of things. In general, all the action takes place at “integer points”, which are points with integer coordinates e.g. (0, 0)
, (1, 2)
, (3, 4)
, etc1. Each integer point occurs at the intersection of two gridlines in the diagram.
In the real problem, the single uncovered point is always found somewhere in a 4×106 by 4×106 unit area, which I’ll call the “search area”. For this example, a small 3x3 search area will suffice. The origin (0, 0)
is always located at the bottom left corner of the search area. I’ll show some area around the search area to give context but remember that we’re looking for uncovered points inside the search area.
In fig. 1 there is a single sensor A
positioned at (2, 2)
with a range of 3
. The area covered by A
is diamond shaped rather than circular. This is because distance is measured by Manhattan distance. Note that some of the sensor’s coverage is outside of the search area, this is okay. In the real problem, we get a text file specifying the position and range of each sensor2. The search area is always 4×106 by 4×106 units.
Finally, there’s a green circle at the point (0, 0)
. This represents the solution to our toy version of the problem (its position of at the origin is just a coincidence). In this case it’s easy enough to see visually that this is the only uncovered point3. Also note that points on the border of a sensor are considered covered.
To drive this all home, let’s look at a slightly more complicated example:
2.1. A Running Example
I’ll use this running example throughout this post:
- We have 4 sensors:
A: position = (0, 4); range = 3
B: position = (4, 3); range = 2
C: position = (1, 0); range = 2
D: position = (4, 1); range = 2
- Sensors are allowed to overlap, as
B
andD
do. - The search area is 4x4 units. Areas at the edge of the search area are in bounds, so we have a total of 52 integer points in our search area, not 42 as you might expect.
- This time the solution is at
(2, 2)
, we can see visually that this is the only integer point in the search area which is uncovered.
Now that we understand the problem, if we look back at fig. 1, my problem data has 40 sensors (given random colours in the visualisation). Somewhere in the 4×106 by 4×106 unit search area there’s a tiny little area that isn’t covered by any of the sensors, you’re not going to be able to see it, it’s far too small. Note that each sensor is comparatively large, with many covering more than 1012 points4. Given the size of the search space, solving real problems by hand isn’t going to cut it, so let’s take a look at the first approach that I tried.
3. Solving via Brute Force
The brute force approach is simple, here’s some Rusty pseudocode:
I’ll use pseudocode throughout this post to summarise the algorithms I discuss. While this pseudocode is quite complete, later on we’ll gloss over a lot of details. This is to focus on the higher-level aspects of the algorithms. You can find my full implementations here.
for x in 0..=4000000 {
'iter_points: for y in 0..=4000000 {
for sensor in sensors {
if manhattan_distance((x, y), sensor.position) <= sensor.range {
continue 'iter_points; // Continue to next point.
}
}
return point; // If we reach this point, we've found the solution.
}
}
Easy… except it takes 3.5 days to run5. Better than solving by hand, but still not good enough.
4. Approach 1: Solving With Column Skipping
With the brute force approach, the primary operation is checking whether a point is covered by any sensors—I’ll call this operation a “coverage check”. We can speed things up by performing fewer coverage checks. Our brute force code scans across each row, column by column. We can improve upon this by taking one large step across a whole sensor each time we hit its left side.
Let’s walk through an example, but first, one more diagramming convention: I’ll use a black dot to indicate points where a coverage check occurs.
- We start our search at
(0, 0)
. - The coverage check at
(0, 0)
fails because(0, 0)
is inA
. - We skip along the row to the first point outside of
A
at(3, 0)
. - The coverage check at
(3, 0)
fails because(3, 0)
is inB
. - Repeating what we did last time, we skip along the row to the first point outside
B
:(6, 0)
. (6, 0)
is out of bounds, we skip the coverage check and move up to the next row at(0, 1)
.- We repeat this process, eventually landing on
(0, 4)
. (0, 4)
passes the coverage check. This is our solution.
4.1. Running Example: Column Skipping
Let’s look at how this plays out with our running example:
4.2. Pseudocode: Column Skipping
And here’s some pseudocode for good measure:
let mut pos = Vec2 { x: 0, y: 0 };
'outer: loop {
for sensor in sensors {
if sensor.pos.manhattan_distance(pos) <= sensor.range {
// Advance x past the sensors range
pos.x = sensor.pos.x + sensor.range - abs(sensor.pos.y - pos.y) + 1;
if pos.x > 4_000_000 {
// Wrap around
pos.x = 0;
pos.y += 1;
}
if pos.y > 4_000_000 {
panic!("No solution found");
}
continue 'outer;
}
}
return pos;
}
4.3. Summary and Performance: Column Skipping
I found this approach surprisingly effective, the runtime has gone from approximately 3.5 days to 199ms on my machine, a factor of 1.5×106 improvement.
Thinking about this critically for a moment: with the brute force method we perform 4×106 coverage checks for each row, now we only need around 2 (based on our worked example). Roughly speaking we’d expect a factor of 2×106 speedup, so in hindsight I shouldn’t have been surprised after all.
I think 199ms is “good enough”, but can we do better?
5. Approach 2: “Range Exclusion”
Honestly, this section is a bit of a slog, and (spoiler alert) it’s only the 2nd best solution. I’d consider skipping to the next section unless you’re really interested. You can always come back later. I found the ideas really interesting, so I didn’t have the heart to cut the section completely.
Now, remember that the problem data I got from AoC only has 40 sensors. That’s miniscule compared to the 4×106 rows in our search area. Perhaps we can find an algorithm which operates primarily on sensors. Even a bad algorithm operating on 40 sensors is likely to be faster than a good algorithm operating on 4×106 points.
After some thought (and cheating6) I managed to find a way to solve the problem by iteratively ruling out sections of the search area. It’s a bit tough to explain, and there are some fiddly details to work out, so let’s start with a simplified version of the problem.
5.1 A Simplified Problem
We can modify the problem slightly to make it much simpler. Let’s forget about Manhattan distances for now; instead, each sensor will cover a rectangular area of the 2D grid. As we’ll see soon, this simplifies the problem because it makes everything axis aligned.
Here’s an example of our simplified version of the problem:
Going back to our goal—to solve the problem by operating primarily on sensors—we can make a couple of observations:
- Focusing on just the rows for now. We saw that the solution is
(2, 2)
, therefore the rowy=2
must not be fully covered by sensors. - Conversely, we know there is a single solution, so every other row must be fully covered. In this case rows
y in {0, 1, 3, 4}
are all fully covered. - Rows may be covered by a single sensor, or a combination of sensors. In this case the rows
y in {0, 1}
are covered by a single sensorD
, the remaining fully covered rowsy in {3, 4}
are covered by the combination of sensors{A, B, C}
. - We can work the other way around too: for any combination of sensors, we can determine which rows they cover. For example:
{D}
covers rowsy in {0, 1}
{A, D}
covers rowsy in {0, 1}
.A
is redundant here, asD
alone has this effect, nonetheless this fact still holds.{A, B, C}
coversy in {3, 4}
{A}
covers no rows{A, B}
covers no rows- … and many more combinations that I’ll skip for brevity cover no rows.
- The rows covered by a set of sensors will always be contiguous.
Based on these observations, we can find all rows which are covered by considering all combinations of sensors. Doing so will leave us with a single remaining row which isn’t covered. The y
point of this row is the y
coordinate of the solution.
Let’s step through this using the example above:
- let
possible_y_values
={0, 1, 2, 3, 4}
. This is all y values in the search area. - all possible combinations of sensors are:
{{A}, {B}, {C}, {D}, {A, B}, {A, C}, {A, D}, and so on...}
(I’ve omitted some for brevity) - Iterate over all sets of sensors, removing covered rows from
possible_y_values
.{A}
covers no rows, no need to updatepossible_y_values
.{B}
covers no rows, no need to updatepossible_y_values
. Okay, from now on I’m going to skip sets of sensors which don’t cover any rows.{A, B, C}
covers{3, 4}
, updatepossible_y_values
from{0, 1, 2, 3, 4}
to{0, 1, 2}
{A, D}
covers rows{0, 1}
. Updatepossible_y_values
from{0, 1, 2}
to{2}
{D}
covers rows{0, 1}
.possible_y_values
remains{2}
- After iterating through all sensor sets, there is only one remaining possible y value:
2
. This is our result for the y coordinate.
We can get the x coordinate by repeating this process for columns instead of rows7, in this case we would also get the value of 2
for x. This leaves us with the final solution of (2, 2)
.
Okay, so we’ve found another way to arrive at our solution, but our goal was to do so while avoiding any sort of iteration over points or rows, did we achieve this? Well, it depends… we’re certainly not performing coverage checks on a per-point or per-row basis anymore, we’re not performing coverage checks at all. However, we are storing rows in possible_y_values
. To make it worse, possible_y_values
contains every row at the beginning of the process. Storing all 4×106 rows seems contradictory to our goal, but all is not lost thanks to an observation we made earlier:
The rows covered by a set of sensors will always be contiguous.
We can take advantage of this by storing ranges of rows. We can store a beginning and end coordinate to represent a range of values. For example, the rows {0, 1, 2, 3, 4, 5}
can be stored as an inclusive range from 0 through 5 which I will denote as [0, 6]
(using the closed-open convention, see Dijkstra’s famous essay). Similarly, the rows {0, 1, 2, 3, 4, 7}
can be stored as a set of ranges {[0, 5], [7, 8]}
8.
Here’s another example using this technique for storing possible_y_values
:
- let
possible_y_values
={[0, 6]}
. - Iterate over all sets of sensors (excluding some for brevity).
{A, B}
covers the range[2, 4]
. Updatepossible_y_values
from[0, 6]
to{[0, 2], [4, 6]}
{B, E, D}
covers the range[5, 6]
. Updatepossible_y_values
from{[0, 2] [4, 6]}
to{[0, 2], [4, 5]}
{A, C}
covers the range[0, 3]
. Updatepossible_y_values
from{[0, 2], [4, 5]}
to{[4, 5]}
- There is only one possible solution:
4
. This is our result for the y coordinate.
Storing possible_y_values
as a set of ranges is a bit more complicated, and in these small examples it doesn’t seem to be worth it. Remember though, we’re covering an area 4×106 units wide with only 40 sensors. The actual ranges we’ll be dealing with are huge. By storing possible_y_values
this way, the total number of values we need to store is a function of the sensor count, rather than the search area size.
Finally, before we move on to the full problem, here is some pseudocode for getting the y coordinate of the solution:
// Initialise the set of ranges `possible_y_values` as a single range covering
// the full search area.
let possible_y_values = {[0, 4000000]};
// Iterate each possible sensor set. Implementation of get_sensor_sets()
// omitted.
for sensor_set in get_sensor_sets() {
// Get the range of rows completely covered by sensor set. This may be an
// empty range. Implementation of get_covered_y_range() is omitted.
let covered_y_range = get_covered_y_range(sensor_set)
// Subtract the covered ranges from possible_y_values. Implementation of
// subtract_range_set() is omitted.
let possible_y_values = subtract_ranges(possible_y_values, covered_y_range)
}
// possible_y_values should only contain a single range at this point. The range
// should only contain one value. This is the solution for y.
return possible_y_values.first().start
At this point the pseudocode is glossing over a lot of details. You can find my full implementations here.
The finer details of how to implement get_sensor_sets()
, get_covered_y_range()
, and subtract_range()
are omitted for brevity. This is admittedly a cop out because they aren’t completely trivial to implement, however I don’t want to distract from the main point, which is the higher-level aspects of the algorithm9.
5.2. Tackling the Full Problem
Now we’ve got that simpler case out of the way, how is it affected when we move onto the full problem? Well, we can use the same approach, however the sensor coverage is no longer axis aligned:
Figuring out which rows are covered visually is a bit more difficult with the unaligned sensor coverage; it’s a lot more difficult to do algorithmically. Instead of trying to tackle this problem, we can shift our perspective to a more convenient coordinate system.
5.2.1. Introducing “Diagonal Space”
Let’s start with a single sensor with range=2
position=(4, 2)
:
We can represent this exact situation in a coordinate system that’s been scaled, translated, and rotated clockwise by 45°, as shown by the axes in fig. 10. For lack of a better name, I’ll call this “diagonal space”, I’ll denote the diagonal space axes as x'
and y'
.
In this coordinate system, the sensor is located at (8, 6)
. Also, if you tilt your head, you’ll see that the sensor coverage is now axis aligned! Here, I’ll do it for you:
In this coordinate system we can mostly reuse the solution from the “simple” version; once we find the solution in diagonal space, we’ll need to convert the solution back into the original coordinate system10. Unfortunately, it’s not all roses, there are still some complications we need to deal with. When we changed coordinate system, we gained axis aligned sensors at the cost of an unaligned search area. This is unfortunate, but I think it’s worthwhile as it allows us to easily operate on sets of sensors. Before tackling the complications, let’s have a look at our go to example.
5.2.2. Running Example in Diagonal Space
We can see that every row in the solution space is completely covered except for the row y'=4
. I’ve annotated the set of sensors which cover each row on the left of the diagram. We see that the approach we took for the simplified problem applies here too.
5.2.3. Complications in Diagonal Space
In the “simple” version, we relied on the observation that “rows covered by a set of sensors are contiguous” to optimise our storage of possible_y_values
. We’ll keep storing possible_y_values
as a set of ranges, however we no longer get a single contiguous set of rows covered by each sensor set. Let’s look at another example to illustrate why:
In this case we see that the sensor A actually covers two contiguous regions, [0, 2]
and [6, 8]
:
We need to account for the possibility of 2 contiguous regions of covered rows. This is a bit more complicated, but it doesn’t change the overall approach.
5.3. Pseudocode: Range Exclusion
Finally, here’s our pseudocode for getting the y coordinate of the solution, it’s very similar to the previous version:
// Convert sensor positions into diagonal space
for sensor in sensors {
sensor.position = rectangular_to_diagonal_space(sensor.position)
}
// Initialise the set of ranges `possible_y_values` as a single range covering
// the full search area.
let possible_y_values = {
[
rectangular_to_diagonal_space(0, 0).y,
rectangular_to_diagonal_space(0, 4000000).y
]
};
// Iterate each possible sensor set. Implementation of get_sensor_sets()
// omitted.
for sensor_set in get_sensor_sets() {
// Get a set of ranges covered by the sensor set. There may be 0, 1 or 2
// returned ranges. Implementation of get_covered_y_ranges() is omitted.
let covered_y_ranges = get_covered_y_ranges(sensor_set);
// Subtract the covered ranges from possible_y_values. Implementation of
// subtract_ranges() is omitted.
let possible_y_values = subtract_ranges(possible_y_values, covered_y_ranges);
}
// Omitted: repeat the same for x...
// possible_y_values and possible_x_values should each contain a single range at
// this point. Due to oddities in converting back and forth from diagonal space,
// the ranges may be up to 3 units large. After converting back to square space
// the resulting area will only contain a single integer point, which is our
// solution.
extract_square_space_solution(possible_y_values, possible_x_values);
We do a bit of converting back and forth from standard to diagonal space and we handle multiple ranges for each sensor set. Otherwise, it’s the same as the previous version.
5.4. Summary and Performance: Range Exclusion
On my machine, the range exclusion implementation takes 4.2ms to run, another 54x speedup, nice! The runtime doesn’t depend on the size of the search space at all, so we could double the search space and the sensor ranges and get the same outcome. Conversely it won’t perform any better on smaller search spaces. It’s a trade of that’s worth it for the particular sensor count and search space that we’re given in this case.
6. Approach 3: “Line Intersection”
We’ve found two very different approaches to solving this problem, but we’re not done yet! There’s another approach we can tackle which is based on taking advantage of the properties of the sensors near the solution11.
6.1. Exploring Patterns Around the Solution
We know our solution is unique, so all integer points surrounding the solution must be covered by a sensor (or outside of the search area if the solution is on a border). Let’s explore this by looking at some example cases, we’ll focus on only the sensors required to cover the 8 coordinates around the solution.
This isn’t an exhaustive list of cases, but it’s enough to gain some intuition for the problem. We see that in all cases the solution is always exactly 1 unit away from the border of a sensor. This makes sense, as we’ve already established that the points adjacent to the solution must be covered.
If we expand each of our sensors’ range by one, then our solution will always be on the border of a sensor, and near the intersection of two borders. Let’s redraw these examples with enlarged sensors:
We see that the solution is indeed always on the border of the sensor. For this analysis, we really only care about the borders of each sensor nearest the solution. Let’s focus on those:
When the borders of sensors intersect at a T or X junction, the intersection is either exactly on an integer point, or it’s exactly halfway between them. We’ll call these intersections “aligned” or “misaligned” respectively.
In the “Box”, “Flower”, and “Splat” cases, the solution lies on an aligned intersection. In the “Hallway” case, the intersection is adjacent to a misaligned intersection. The solution is always either on an aligned intersection or adjacent to a misaligned one.
Based on this, we can find solutions by following these steps:
- Enlarge the sensors’ ranges by 1.
- Find all intersections between the borders of the enlarged sensors.
- At each aligned intersection, perform a coverage check.
- For each misaligned intersection, perform a coverage check on each of the 4 adjacent integer points.
The number of intersections is a function of the position and number of sensors—but not the size of the search area—so this will scale well as the search area grows.
6.2. Running Example: Line Intersection
Let’s work through our running sample problem:
We see that the solution at (2, 2)
is on an aligned intersection and will be picked up by a coverage check.
6.3. Corner and Edge Cases
Now we have some literal edge cases to take care of, let’s look at some examples where the solution lies on the edge of the search area:
Fortunately, our assumptions still hold for edge cases. How about corner cases?
Nope! Fortunately, there are only 4 corners, so we can simply hardcode a coverage check for each corner, and we’re done!
6.4. Pseudocode: Line Intersection
Here’s the final pseudocode:
// Initialise points_to_check with the 4 corners of the search area.
let points_to_check = [(0, 0), (0, 4000000), (4000000, 0), (4000000, 4000000)]
for intersection_point in get_intersection_points() {
if is_aligned(intersection_point) {
points_to_check.add(intersection_point);
} else {
x = intersection_point.x;
y = intersection_point.y;
points_to_check.add((x - 1, y));
points_to_check.add((x + 1, y));
points_to_check.add((x, y - 1));
points_to_check.add((x, y + 1));
}
}
for point in points_to_check {
for sensor in sensors {
if manhattan_distance(point, sensor.position) <= sensor.range {
break; // Continue to next point.
}
}
return point // If we reach this point, we've found the solution.
}
6.5. Summary and Performance: Line Intersection
This solution runs in 8.6µs, another 488x speedup! Similar to the Range Exclusion approach, the runtime only depends on the number of sensors.
7. Performance Comparison
Now that we’ve reduced our runtime from days to microseconds, let’s compare the approaches:
This plot uses a log scale, because otherwise the chart would be completely dominated by the brute force approach. Each horizontal line represents a 10x increase in runtime.
I mentioned that the runtime of these approaches depends on the number of sensors, and for brute force and range exclusion, also the size of the search area. Let’s look at each approach when run on the smaller example data that AOC provides:
The example has a 20x20 search area and 14 sensors. We see that the complexity of the range exclusion approach isn’t worth it at this scale, performing worse than even the brute force approach. The column skipping approach really shines here12
8. Closing Thoughts
At the time of writing, I’ve only completed AoC 2022 days 1 through 15, but this is definitely my favourite problem so far. I had a ton of fun exploring the different approaches and measuring their performance. I think the size of the search area combined with the use of Manhattan distance13 makes the problem shine. The most basic approach—brute force—is only just too slow14, but any optimisation at all allows us to solve the problem in reasonable time, opening the door to many different approaches.
9. Final Visualisations
And finally, here’s some visualisations of the final result for my problem data15:
If you skipped section 5, now is the time to go back and have a look… or y’know, go live your life, I understand!
Because everything takes place on integer points, you could say the problem takes place in a point lattice. The integer points we’ll be talking about throughout are also known as lattice points. ↩︎
I’m oversimplifying this a little, but in essence this is true. ↩︎
Technically there’s a small triangular area which isn’t covered containing an infinite number of points, but remember we’re only really interested in what happens at integer points. ↩︎
A sensor with range R covers (R+1)2 + R2 integer points. e.g. a sensor with
range=0
covers 1 coordinate, a sensor withrange=3
covers 25. This formula is a little more complicated than you might expect because the square area is not axis aligned. For some intuition behind this formula, have a look at this sensor withrange=2
. It can be broken down into a 3x3 grid (red) and a 2x2 grid (blue).Tangentially related: Pick’s Theorem relates the area of a polygon with the count of its internal and border lattice points (see footnote 1). ↩︎On my machine I can check 5.3×107 points/sec. Extrapolating for our search area containing 1.6×1013 integer points gives an estimated run time of 3.5 days in the worst case. ↩︎
Credit where credits due: I couldn’t think of anything until I skimmed the Reddit solution thread. I was trying to skim for inspiration without spoiling for myself. Luckily I saw a mention of rotating the coordinate space (you’ll see why this is relevant later) which got me unstuck. ↩︎
After finding the solution for y, we can shortcut the x coordinate. Now that we know the exact y value for the solution, it’s sufficient to consider sensors one at a time instead of sets of sensors. ↩︎
In my implementation I store a set of ranges as a dynamic array (Rust Vec) of integer 2-tuples. I’m sure I could use a more efficient data structure. Perhaps an interval tree, I’m not sure? ↩︎
One important detail which I gloss over is that I skip checking a large number of sensor sets. In particular those where the sensors don’t form a contiguous area, and those where the intersection of the sensor’s y values is empty. These sensor sets can’t cover any rows that aren’t already covered by some other sensor set. Doing this dramatically cuts down the number of sensor set checks. This is actually critical because there are 240 - 1 non empty sets of sensors, the vast majority of which are redundant. ↩︎
In fig. 10. I’ve set up the origin of the “diagonal space” coordinate system so that
y'
andx'
touch the top left and bottom left corners of the solution space respectively. There’s nothing special about this convention, I chose it to make the visualisations neater. Nonetheless, here’s some Rust code for converting between the systems using this convention:// `dimension` is the width of the search area. fn rectangular_to_diagonal(vec: &Vec2, dimension: i32) -> Vec2 { Vec2 { x: vec.x - vec.y + dimension, y: vec.x + vec.y, } } fn diagonal_to_rectangular(vec: &Vec2, dimension: i32) -> Vec2 { Vec2 { x: (-dimension + vec.x + vec.y) / 2, y: (dimension - vec.x + vec.y) / 2, } }
A simpler convention would leave the origin at the bottom left corner of the search area in both coordinate systems. This would make the conversion functions simpler and remove the dependency on
dimension
for conversions. ↩︎Once again, credit to the Reddit solution thread. Skimming this thread, I saw a couple of references to line intersections which got me started on this approach. ↩︎
To get a proper idea, I should really measure more cases. For example, all implementations exit as soon as they find the solution, so there is a bit of a luck factor involved as well. It’d also be nice to try a few more combinations of search area sizes and sensor counts. ↩︎
If Euclidian distance were used instead, we’d need to deal circular sensor coverage.
Additionally, using Manhattan distance means that the distance between any two integer points is always an integer, keeping the solutions free from floating point complications. For those who’ve been following along in the footnotes: this is thanks to the fact that the lattice is closed under addition and Manhattan distances are calculated soley with addition16. ↩︎
Actually, some people did brute force it on a GPU, which is pretty cool. ↩︎
In my case it’s a “Box” type solution (see fig. 16). I’m not sure if this is the case for everyone. ↩︎
Just for fun, here’s another interesting distance metric that I came across while writing this: Chebyshev distance17. ↩︎
What’s that? Nested footnotes are considered bad style? Well, I never… ↩︎
Comments
Thanks for reading! Feel free to leave a comment below, comments are linked directly to this GitHub discussion, so you can comment there directly if you prefer.