Contents

Advent of Code: 2020 Day 02 solutions

SPOILER ALERT This is a post with my solutions and learnings 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.

My programming language of choice is python and all examples below are in python.

Key learnings

  • String handling

This puzzle teaches you to handle strings. They are handled differently in every programming language and it’s valuable to get comfortable with them. I find that python makes it easy to work with strings, but in other languages it might require more getting used to. You are going to need string handling a lot during Advent of Code as the input file is most often a text-file.

Puzzle part 1

The puzzle is to count the number of valid passwords in your input according to a policy.

Example input:

1-3 a: abcde
1-3 b: cdefg
2-9 c: ccccccccc

Each line gives the password policy and then the password. The password policy indicates the lowest and highest number of times a given letter must appear for the password to be valid. For example, 1-3 a means that the password must contain a at least 1 time and at most 3 times.

Parse input

First step is to save the input in a local file and parse it in python:

# Open the input file
inputfile = open('02.input', 'r')

# Parse lines 
data = [x.strip() for x in inputfile.readlines()]

Solution

def part1(data):
    total = 0
    for x in data:
        span, letter, password = x.split()
        start, end = map(int, span.split('-'))

        count = password.count(letter[0])

        if start <= count and count <= end:
            total +=1
    return total

result = part1(data)
print("Result part 1: %d" % result)

Part 2

The second part has the same input, but the policy is changed.

Each policy actually describes two positions in the password, where 1 means the first character, 2 means the second character, and so on. (Be careful; Toboggan Corporate Policies have no concept of “index zero”!) Exactly one of these positions must contain the given letter. Other occurrences of the letter are irrelevant for the purposes of policy enforcement.

Solution

def part2(data):
    total = 0
    for x in data:
        span, letter, password = x.split()
        start, end = map(int, span.split('-'))

        password1 = password[start-1] == letter[0]
        password2 = password[end-1] == letter[0]
        
        # Count if either first or the other character is correct. (xor)
        if password1 != password2:
            total +=1
    return total

Alternative solutions

Functional style

Going for a more functional style can make the code a bit cleaner. E.g there are no mutable variables or loops in this case. Though it requires the programmer to be comfortable with filter, map, reduce functions and so on.

def validate(line):
  span, letter, password = x.split()
  start, end = map(int, span.split('-'))
  count = password.count(letter[0])
  return start <= count and count <= end

def part1(data):
  valid_passwords = filter(validate, data) 
  return len(valid_passwords)

Regex

Regex is a powerful tool to use instead of splitting the strings. In python I find it easier to split strings as I did above. But in some cases (or other languages) it can be helpful to use regex instead. Here is an example of how the validation function could look like:

import re
def validate(line):
    start, end, letter, pw = re.search(r'(\d+)-(\d+) (\w): (\w+)', line).groups()
    return int(start) <= pw.count(letter) <= int(end)

Python explanations

Splitting strings

As all lines have the same format 1-3 a: password I know that splitting on space them will always output an array with 3 elements. In python we can automatically destruct it:

span, letter, password = x.split()

# Is same as:
output = x.split(' ')
span = output[0]
letter = output[1]
password = output[2]

Parsing numbers

The map function takes two parameters; a transform-function and an array. By using int as transform-function it will transform the strings to an integer if it’s parseable.

start, end = map(int, span.split('-'))

# Is same as:
a, b = span.split('-')
start = int(a)
end = int(b)

Counting occurrences

Python has a built-in function count on strings. It takes an string as parameter and it will return the number of occurrences of that string.

s = 'aabbccaabb'
count = s.count('a')
# count: 4

Character position in string

s = 'aabcd'
index = 2
char = s[index]
# char: b