Advent of Code: 2020 Day 09 solutions
⚠️ SPOILER ALERT ⚠️ This is a post with my solutions and learning from the puzzle. Don’t continue reading if you haven’t tried the puzzle on your own yet.
If you want to do the puzzle, visit adventofcode.com/2020/day/9.
My programming language of choice is python
and all examples below are in python.
Key learning
- Window sliding
This puzzle teaches us to slide through an list of values using a window. It is possible to solve with nested for-loops and if-statements. But it gets much easier utilizing a window to operate on.
Puzzle
The puzzle is to validate number in a list. A number is valid if it is the sum of any of the 25 immediately previous numbers. All first 25 numbers are counted as valid. The input is a file with one number per line.
Example input:
35
20
15
25
47
40
62
55
65
95
102
117
150
182
127
219
299
277
309
576
Part 1
In part 1 we have to find the first number that is NOT valid.
Parsing input
# Parse directly into ints.
numbers = [
int(line.strip())
for line
in open('09.input', 'r').readlines()
]
Solution (bad)
The first concept is to figure out how to traverse the list and keep track of the 25 previous number. A naive solution is to iterate back to all previous numbers and try out all possible combinations of sums. Here is a example of that naive thinking. I don’t even want to think about the time complexity. Though it still solves the puzzle on my computer in an instant.
def part1(numbers):
# Iterate all numbers
for i, n in enumerate(numbers):
# Skip "preamble"
if i < 25:
continue
# Iterate all 25 previous numbers
is_valid = False
for j in range(i-25,i):
# Try out all combinations to sum to n
for k in range(i-25, i):
if j != k:
if numbers[j] + numbers[k] == n:
is_valid = True
break
if is_valid:
break
if is_valid:
break
# If is not valid, then we found our answer
if not is_valid:
return n
print "Solution part 1: %d" % part1(numbers)
Solution (good)
Using the concept of a sliding window
we can create a more performant and clean
solution. The sliding window is a list containing the previous 25 elements.
After each iteration we append the current number at the end and pop the first element to keep the size consistent of 25 elements.
We can also improve our validation. One way is to calculate the differences between
the values in the window with the current number n
. If the lists of differences
intersects with the window, then there are two values that sum up to n
. Note:
this assumes that it DOES NOT exist numbers that are half of n
. Works on my input, might
not work on yours.
from collections import deque
def part1(numbers):
window = numbers[:25]
for n in numbers[25:]:
differences = set([n - x for x in window])
is_valid = len(window.intersection(differences))
if not is_valid:
return n # Found answer
else:
window.append(n) # Append n at end of list
window.pop(0) # Remove first element of list
print "Solution part1: %d" % part1(numbers)
Solution (optimized)
If we think about performance it cost O(N)
to pop the first element in lists.
Using a deque
it lowers the time-complexity. Deque:
appends and pops from either side of the deque with approximately the same O(1) performance in either direction
Deque also has a maxlen
parameter which ensures that the window stays at a certain size.
To optimize the validation we could reuse the code from Day 01
. Essentially it just
gives us a early return once a valid number is found with worst case O(N)
time complexity. The previous example requires
parsing the window once and then doing an intersection. Set intersection has worst case of O(N^2)
which looks slower. But the window size is set to 25, which makes the worst case O(500)
which simplifies to O(1)
. So both variants have equal time complexity.
Here is the code:
from collections import deque
def valid(arr, n):
differences = set()
for x in arr:
if n - x in differences:
return True
differences.add(x)
return False
def part1(numbers):
window = deque(numbers[:25], maxlen=25)
for n in numbers[25:]:
if not valid(window, n):
return n
window.append(n)
print "Solution part1: %d" % part1(numbers)
Part 2
In part 2 we use the invalid number we find in part 1.
We have now to find a contigous set of at least two numbers which sum to that invalid number
. The answer is the sum of the largest and smallest number in that
set.
Using the learning from part 1 we can jump into the clean solution using a sliding window
. While iterating the list of numbers we push the current n
to the window. Then if the window-sum is too big we pop the first element. And continue doing so as until the window-sum is equal to our invalid number, or smaller. We can keep track of the window-sum in a variable.
Solution
def part2(numbers, needle):
window = deque()
window_sum = 0 # Sum for numbers currently in window
for n in numbers:
window.append(n)
window_sum += n
while window_sum > needle: # Pop head until window_sum is within limit
window_sum -= window.popleft()
if window_sum == needle: # Result found
return max(window) + min(window) # Sum the smallest and largest
solution_part_1 = part1(numbers)
print "Solution part 2: %d " part2(numbers, solution_part_1)
Thanks for reading!
I hope these solutions were insightful for you.
Complete code can be found at: github.com/cNille/AdventOfCode/blob/master/2020/09.py