Takeaways from Advent of Code 2024
On December 1st I was scrolling Hacker News when I came across a post for the Advent of Code. I had been wanting to get back into software so I decided to check it out and give it a try.
While I first started coding in middle school, and was at one point going to major in Computer Science, I eventually switched to Electrical Engineering. While I had kept coding in some jobs and personal projects, I've spent the last year as a DevOps engineer, creating pipelines and scripts but not doing the coding problems that I had been used to. I have a few personal projects on the horizon, and I felt that Advent of Code would be a good way to try to reinvigorate my love of software while freshening up my Python. Plus, at just one problem a day, I thought it would be fairly low-commitment, and easy to maintain.
I didn't use an obscure language or come from a non-coding background, but I thought I would share my experience.
My Experience
Overall I found Advent of Code to be highly enjoyable. The goofy stories were fun to read, and I greatly enjoyed how you had to frequently determine the abstractions to model the problems instead of simply writing an algorithm for something abstract (like LeetCode problems often are, for example). While I don't frequent Reddit, the community discussion there makes it feel like you're part of something.
Because of both business and personal travel for the holidays I did my Advent of Code across 3 different timezones (Qatar, Eastern, and Mountain). I'm going to blame that on my lack of placing on the leaderboard (as if I would have been fast enough on any of the problems).
The commitment level ended up being a little greater than I had been expecting, though that may have partly been due to how rusty I was at programming. While my code definitely improved over the course of the month I finally hit the "wall" that people talk about, obsessing over a bug in my code for a full day. Overall, however, most days I was able to push through very quickly, and the constant repetition is wonderful for picking up a new skill, or refreshing a rusty one. This sort of daily ritual is something that I will try to keep in my life.
Here I'm going to discuss my experience with some of my favorite problems with this year's Advent of Code. All of the code is available in my GitHub repo.
1. Day 6 - Guard Gallivant
Day 6 was the first day I had trouble. Part 1 came very quickly, but part 2 was very difficult for me. To this day it is the only one I have not optimized enough to run in under 10 seconds - it takes 40. (I have one idea to try, but I don't know if it will make it fast enough.)
Day 6 is where I've gained much appreciation for the community around the Advent of Code. When I was stuck, I found someone on Reddit who had completed it and read their code on GitHub. They coded in an OOP style that was very easy to read, and I quickly found my issue1.
I was able to fix my code, but seeing his OOP design inspired me to re-write an OOP approach of that problem of my own. Similarly, while on Reddit I saw an animation. Looking at their code I learned about printing escape sequences to move a cursor around a terminal screen which allowed you to re-write terminal values. I added this to my OOP approach as well.
I love how people in the community are able to inspire and uplift each other like this.
2. Day 11 - Caching is King
The one more 'advanced' technique that was frequently used amongst a vast swath of the problems was that of caching - at the end of a function store the result in a hash table (Python's dict()).
Day's 11, 19, and 21 were all made fairly easy once memoization was applied (though day 21 was certainly a bit more complicated by the others).  While the technique is simple, I always find it extraordinary how adding a dict can reduce the computation time from several years to less than a second, and changing a recursive foo() + foo() to 2 * foo() can change the number of function calls from 2^50 to 50.
3. Day 15 - Recursion is Unreasonably Effective
Recursive solutions kept coming so naturally to me throughout this month. I think that doing some inductive proofs in my math courses have helped a lot. It's incredibly useful to 'prove' recursive ideas to yourself with inductive reasoning, and I'd recommend researching a few inductive proofs to anyone who struggles with recursive code.
That said, there were times when I over-used it. In day 20, when I wanted to enumerate the track, I started with a recursive solution that quickly created a stack overflow, when a simple iterative solution worked perfectly:
# Changes every area on the map track to an int representing 
#   that position in the track
def enumerate_track(start, race_map):
  sym = "."
  cur = start
  num = 0
  while sym != "E":
    race_map[cur.y][cur.x] = num
    num = num + 1
    cur, sym = get_next_pos(cur, race_map)
  race_map[cur.y][cur.x] = num
  returnWhen applied appropriately, though, recursion can be wonderful. In day 15 I combined recursion with the idea of "General Purpose code" as described in A Philosophy of Software Design (another thing I found from Hacker News). I came up with a single function that can handle boxes that are both 1 and 2 blocks wide, and look ahead without updating so both no boxes will be updated if only one of the paths has a wall.
def move_box(pos, dir, map, move_half=True, update=True):
  # Move the other half of the box, if necessary
  if is_vertical_move(dir) and move_half:  
    if is_left_edge_of_box(pos, map):
      if not move_box(new_pos(pos, ">"), dir, map, move_half=False, update=update):
        return False
    elif is_right_edge_of_box(pos, map):
      if not move_box(new_pos(pos, "<"), dir, map, move_half=False, update=update):
        return False
  # Cannot update if there is a wall
  if is_wall(new_pos(pos, dir), map):
    return False
  # If there is a box in the space that this box is being moved to, move it
  if is_box(new_pos(pos, dir), map):
    if not move_box(new_pos(pos, dir), dir, map, update=update):
      return False
  # Move the box
  if update:
    set_map(new_pos(pos, dir), get_sym(pos, map), map)
    set_map(pos, ".", map)
  return True4. Day 24 - The 'Wall'
This was my first year hearing about and doing Advent of Code, and in the early days of the month I read a lot of comments about "the wall" - one day that is exceptionally difficult for nearly everyone. Day 24 was my wall. Though I also found it the most enjoyable.
Part 1 was again fairly simple - place some values on some wires and simulate some wire AND and XOR operations. Part 2 was where things got interesting. Simply put, all of the wires were supposed to simulate the addition of binary numbers, but 4 of the wires were swapped. You have to find those wires.
There was a lot of boilerplate that was required for testing the additions - setting values to the wires, simulating the wires, reading values from the wires. But much of it was extremely open-ended, and was not the type of problem you could use a generic algorithm off the internet to solve.
For one, you could analyze the wire structure and analytically determine which are out of place. You may not even need the testing structure in that case. On the other hand, you could randomly switch every combination of 4 wires and test the addition - though this would almost surely take too long to run. While I liked the idea of analyzing the topology of the wires, I wasn't sure how to implement that, so I tried to narrow down the search possibilities.
I created a directed graph from the wire inputs, but looking forward from the inputs shows all later outputs. Makes sense, as the carry signal propagates through the entire circuit. But, unexpectedly I saw that the first output of every input was at the same number, eg x0 was connected to z0, x1 and x1, and so on. I was expecting 8 to not match, for the 4 swaps. This made the problem trickier, as the 4 swaps were all 'intra-signal', swapped connections between the paths of x5, y5, and x5, for example.
To approach this I created another directed graph of the reverse connections.  By interesting the paths of (xn union yn) and (zn union z(n+1)) you can find the wires responsible solely for that addition and carry.  Now when we find a problem for a specific wire group we can try to swap only those wires until our addition works.
To implement this we create a small set of test additions and iterate over the wire groups and combinations, and we have our answer.
In Summary
I found the Advent of Code very fun and rewarding.  I also learned so much from revisiting Python and dedicating some time to these challenges.  It was so nice to invest some time into learning some of the basic libraries like collections, how to type my code properly, and other basic python principles that I may have known but haven't used in years.
It's also nice to see myself improve. I remember, in high school, writing a sudoku solver in a week or two. This week I wrote a brand new sudoku solver in Python in an hour or two. I think Advent of Code is a fantastic way to both stay fresh or improve your code. Maybe next year I'll try to complete it in a functional programming language, something I don't have as much familiarity with.
Footnotes
- 
I had misunderstood the question. It's somewhat technical, but for every square that the guard moved into I was testing whether placing a block there would cause the guard to enter a loop. The actual question was which squares can you place a block that will cause the guard to enter a loop. This is different because if a guard visits the same square twice, but there is a block there, he will not approach that block from the second direction. ↩