Solution in Python for the day 5 puzzle of the 2021 edition of the Advent of Code annual programming challenge.
They tend to form in lines; the submarine helpfully produces a list of nearby lines of vents (your puzzle input) for you to review. For example:
0,9 -> 5,9 8,0 -> 0,8 9,4 -> 3,4 2,2 -> 2,1 7,0 -> 7,4 6,4 -> 2,0 0,9 -> 2,9 3,4 -> 1,4 0,0 -> 8,8 5,5 -> 8,2
The input contents are in form of int,int -> int,int
.
Each line of vents is given as a line segment in the format x1,y1 -> x2,y2 where x1,y1 are the coordinates of one end the line segment and x2,y2 are the coordinates of the other end. These line segments include the points at both ends. In other words:
- An entry like 1,1 -> 1,3 covers points 1,1, 1,2, and 1,3.
- An entry like 9,7 -> 7,7 covers points 9,7, 8,7, and 7,7.
For now, only consider horizontal and vertical lines: lines where either x1 = x2 or y1 = y2.
For now. guess what is part two.
So, the horizontal and vertical lines from the above list would produce the following diagram:
.......1.. ..1....1.. ..1....1.. .......1.. .112111211 .......... .......... .......... .......... 222111....
Such a diagram will also be handy for debug purposes.
In this diagram, the top left corner is 0,0 and the bottom right corner is 9,9. Each position is shown as the number of lines which cover that point or
.
if no line covers that point. The top-left pair of1
s, for example, comes from 2,2 -> 2,1; the very bottom row is formed by the overlapping lines 0,9 -> 5,9 and 0,9 -> 2,9.To avoid the most dangerous areas, you need to determine the number of points where at least two lines overlap. In the above example, this is anywhere in the diagram with a 2 or larger - a total of 5 points.
Consider only horizontal and vertical lines. At how many points do at least two lines overlap?
The input contents are composed of number of independent lines, which was also the case for the prior puzzles of 2021.
217,490 -> 217,764
44,270 -> 373,599
440,139 -> 440,303
161,663 -> 345,663
The goal is to obtain an efficient structure for storing these data, such as:
((217, 490), (217, 764))
((44, 270), (373, 599))
((440, 139), (440, 303))
((161, 663), (345, 663))
Also like for prior puzzles, the contents is handled on a per-line basis using readlines()
.
lines = iter(open(filename).readlines())
Each single line is composed of the following tokens, separated by a whitespace character.
Position | Type | Regex |
---|---|---|
#0 | integer | \d+ |
#1 | integer | \d+ |
#2 | string | -> |
#3 | integer | \d+ |
#4 | integer | \d+ |
Parsing these tokens can be done a number of ways:
- regex matching:
- match the complete line against a regex
- apply a string to int conversion on all four integers
import re
...
REGEX = r'(\d+),(\d+) -> (\d+),(\d+)'
...
for line in iter(open(filename).readlines()):
m = re.search(pattern=REGEX, string=line)
integers = [int(t) for t in m.groups()]
yield tuple(integers[0:2]), tuple(integers[2:4])
- tokenization:
- substitute comas by whitespaces
- split the string between whitespaces
- discarding the token
->
at position #2 - apply a string to int conversion on all four integers
for line in iter(open(filename).readlines()):
tokens = line.strip().replace(',', ' ').split(' ')
tokens.pop(2)
integers = [int(t) for t in tokens]
yield tuple(integers[0:2]), tuple(integers[2:4])
Comparing the runtime of both implementations, the solution relying on split()
takes around 30 % less time. The complete method being:
def load_contents_token(filename: Path) -> tuple[tuple, tuple]:
"""Load and convert contents from file
:param filename: input filename
:return: coordinates
"""
for line in iter(open(filename).readlines()):
tokens = line.strip().replace(',', ' ').split(' ')
tokens.pop(2)
integers = [int(t) for t in tokens]
yield tuple(integers[0:2]), tuple(integers[2:4])
Running pylint
on the script yielded a number of issues, some of which required changing the load_contents()
method:
def load_contents(filename: Path) -> tuple[tuple, tuple]:
"""Load and convert contents from file
:param filename: input filename
:return: coordinates
"""
with open(filename, encoding='UTF-8') as buffer:
for line in iter(buffer.readlines()):
tokens = line.strip().replace(',', ' ').split(' ')
tokens.pop(2)
integers = [int(t) for t in tokens]
yield tuple(integers[0:2]), tuple(integers[2:4])
One of the most important aspects of software design is selecting an appropriate structure for storing data. For instance the problem deals with a number of points which can be crossed by a number of segments. This requires a direct access to arbitrary points, thus calling for a map-like structure which is the dict
in Python with the key being the coordinates and the value being the number of crossed segments.
Before going any further we can already write a debug method drawing the diagram as shown in the puzzle statement.
def draw_diagram(coordinates: dict) -> None:
"""Draw diagram of each coordinates
:param coordinates: map of points
:return: nothing
"""
points = coordinates.keys()
start_col = min(col for col, row in points)
end_col = max(col for col, row in points)
start_row = min(row for col, row in points)
end_row = max(row for col, row in points)
for row in range(start_row, 1 + end_row):
line = ''
for col in range(start_col, 1 + end_col):
if (col, row) not in points:
line += '.'
else:
line += str(coordinates[(col, row)])
print(line)
The puzzle states that each set of coordinate represents a vertical or horizontal line. Implementing this processing requires locating the common axis and walking the points between the line ends.
def solve_part_one(contents: any) -> int:
"""Solve the first part of the challenge
:param contents: input puzzle contents
:return: expected challenge answer
"""
coordinates = defaultdict(int)
for segment in contents:
horizontal_segment = segment[0][0] == segment[1][0]
vertical_segment = segment[0][1] == segment[1][1]
if horizontal_segment:
start_col = min(segment[0][1], segment[1][1])
end_col = max(segment[0][1], segment[1][1])
for col in range(start_col, 1 + end_col):
point = (segment[0][0], col)
coordinates[point] += 1
elif vertical_segment:
start_row = min(segment[0][0], segment[1][0])
end_row = max(segment[0][0], segment[1][0])
for row in range(start_row, 1 + end_row):
point = (row, segment[0][1])
coordinates[point] += 1
else:
continue
#draw_diagram(coordinates=coordinates)
overlaps = Counter(coordinates)
answer = sum(1 for i in list(overlaps.values()) if i >= 2)
return answer
Wasn't happy with the execution time. Replacing lookups to the segment
variable in the hot paths (i.e inner nested for
loop) shaved nearly 15 % of the runtime.
def solve_part_one(contents: any) -> int:
"""Solve the first part of the challenge
:param contents: input puzzle contents
:return: expected challenge answer
"""
coordinates = defaultdict(int)
for segment in contents:
horizontal_segment = segment[0][0] == segment[1][0]
vertical_segment = not horizontal_segment and segment[0][1] == segment[1][1]
if horizontal_segment:
start_col = min(segment[0][1], segment[1][1])
end_col = max(segment[0][1], segment[1][1])
x = segment[0][0]
for col in range(start_col, 1 + end_col):
coordinates[(x, col)] += 1
elif vertical_segment:
start_row = min(segment[0][0], segment[1][0])
end_row = max(segment[0][0], segment[1][0])
y = segment[0][1]
for row in range(start_row, 1 + end_row):
coordinates[(row, y)] += 1
#draw_diagram(coordinates=coordinates)
overlaps = Counter(coordinates)
answer = sum(1 for i in list(overlaps.values()) if i >= 2)
return answer
Checking the byte-code we clearly see that performing the x = segment[0][0]
and y = segment[0][1]
yields differences in the byte-code. For the latter operation:
LOAD_FAST 1 (coordinates)
LOAD_FAST 12 (row)
-LOAD_FAST 2 (segment)
-LOAD_CONST 1 (0)
-BINARY_SUBSCR
-LOAD_CONST 2 (1)
+LOAD_FAST 11 (y)
BINARY_SUBSCR
BUILD_TUPLE 2
DUP_TOP_TWO
BINARY_SUBSCR
LOAD_CONST 2 (1)
INPLACE_ADD
ROT_THREE
STORE_SUBSCR
JUMP_ABSOLUTE 130 (to 260)
JUMP_ABSOLUTE 6 (to 12)
Contents | Command | Answer | Time |
---|---|---|---|
input.txt |
./day_5.py input.txt -p 1 |
6841 |
Unfortunately, considering only horizontal and vertical lines doesn't give you the full picture; you need to also consider diagonal lines.
Saw this one coming.
Because of the limits of the hydrothermal vent mapping system, the lines in your list will only ever be horizontal, vertical, or a diagonal line at exactly 45 degrees. In other words:
To be honest the 1:1 diagonal ratio simplifies things quite a bit.
Considering all lines from the above example would now produce the following diagram:
1.1....11. .111...2.. ..2.1.111. ...1.2.2.. .112313211 ...1.2.... ..1...1... .1.....1.. 1.......1. 222111....
You still need to determine the number of points where at least two lines overlap. In the above example, this is still anywhere in the diagram with a 2 or larger - now a total of 12 points.
Consider all of the lines. At how many points do at least two lines overlap?
The second part consisted in adding handling for segments which are neither horizontal nor vertical.
Diagonal segments may come in two slopes and two directions, requiring handling all these four cases.
def solve_part_two(contents: any) -> int:
"""Solve the second part of the challenge
:param contents: input puzzle contents
:return: expected challenge answer
"""
coordinates = defaultdict(int)
for segment in contents:
horizontal_segment = segment[0][0] == segment[1][0]
vertical_segment = not horizontal_segment and segment[0][1] == segment[1][1]
if horizontal_segment:
start_col = min(segment[0][1], segment[1][1])
end_col = max(segment[0][1], segment[1][1])
x = segment[0][0]
for col in range(start_col, 1 + end_col):
coordinates[(x, col)] += 1
elif vertical_segment:
start_row = min(segment[0][0], segment[1][0])
end_row = max(segment[0][0], segment[1][0])
y = segment[0][1]
for row in range(start_row, 1 + end_row):
coordinates[(row, y)] += 1
else: # diagonal segment
x1 = segment[0][0]
y1 = segment[0][1]
x2 = segment[1][0]
y2 = segment[1][1]
inc_col = 1 if x2 > x1 else -1
inc_row = 1 if y2 > y1 else -1
for i, _ in enumerate(range(abs(x2 - x1) + 1)):
coordinates[(x1 + inc_col * i, y1 + inc_row * i)] += 1
#draw_diagram(coordinates=coordinates)
overlaps = Counter(coordinates)
answer = sum(1 for i in list(overlaps.values()) if i >= 2)
return answer
Contents | Command | Answer | Time |
---|---|---|---|
input.txt |
./day_5.py input.txt -p 2 |
19258 |
1178.9 ms |