Skip to content

Latest commit

 

History

History
298 lines (251 loc) · 17.2 KB

README.md

File metadata and controls

298 lines (251 loc) · 17.2 KB

Advent of Code 2024

switching around between python, go and ocaml - sometimes all of them!

Day 1

Did this one in Python and in Ocaml - it was straightforward reading in the data and flipping it around into the right shape to do the calculations it was asking for over the lists. Sorted them both for part 1, for part 2 used a hashtable to count the occurrences of the number in the right hand list. Added go for funsies - haven't done any go in a while.

Day 2

Python and Ocaml again! Python for speed, Ocaml for fun. I just implemented the level checks that it asked for in part 1 and counted the valid levels. In part 2 I realized that the levels (rows) were not very long so it wasn't a big deal to just brute force and try them all instead of trying to find a better way to find out of the level can be made valid. Had a bit of trouble getting it going in Ocaml - mostly with getting the diff list to work (just use mapi instead of map2)

EDIT: An example of the linear time solution in Python on reddit: https://www.reddit.com/r/adventofcode/comments/1h4ncyr/comment/m00dpfi/

EDIT2: I also added a Go solution for day 2

Day 3

Part 1 in 4 minutes in Python! This was regex to grab the numbers and then just multiply and sum them.

Part 2 was a different beast - I made a mistake with the new lines in the file, I was processing line by line and resetting the ranges for the do/don't switch with each line instead of keeping it going for the whole file - took me like an hour to figure out what I was doing wrong. I did the same regex to get the numbers, and to also get the do() and don't() switches, tracked the indices of the switches and only applied the mults when the switch was on. Cleaning up and adding an Ocaml solution tomorrow am!

EDIT: added part 1 Ocaml solution before work - part 2 will take some more thinking (need to figure out how to do it functionally instead of imperatively)

EDIT2: cleaned up python part 2 more - thinking about how to do it in ocaml gave me a better idea of how to do in general I think, tbd

Day 4

Scanning around a grid to look for words (it's a word search puzzle). At first I thought it would be really fast to just scan left/right, up/down, and then use the diagonal traverse to check for diagonal - it was running but not quite right (I know now I was off by about 20 in my real input so I must have missed a case). Switched it to check all directions when it finds an 'X' (part 1), then the same idea for part 2 (but with 'A' instead of 'X', and in the middle instead of in the start)

EDIT: Added Ocaml solution at lunchtime to get some more practice. Basically the same as the python solution, but figuring out the recursion traversal was a bit tricky for me, I was double counting / revisting in the early versions. Otherwise it's the same, look at each spot, make the word that you find in each direction starting there (part 1) or centered there (part 2), and then check if it's the word we're looking for.

Day 5

Part 1 was checking the conditions to determine which 'updates' were in the right order - I just looped through and checked them. Part 2 was to sort them based on the conditions given - I started to implement a topo sort, but noticed that that updates are all relatively small and had a hunch that I was overthinking it (again) - so I just sorted them by swapping elements when the condition was not met, and then repeating until it was sorted, I think it's kind of bubble sort. In Ocaml for part 2 - I put the conditions into a hashtable to make it easier to use them to check if the condition was met, and then sorted using the built-in sort function and a custom compare function (checking the conditions). I was originally trying to do that in python too, but I didn't remember how to use a custom compare function (not just key function) but I get it now - might re-write it to use that in python too (need to use cmp_to_key from functools)

EDIT: reworked python to use cmp_to_key and a custom compare function for the sort - it's a lot cleaner and ~5x faster

Day 6

Part 1 was walking on the grid until an obstacle was hit, and then turning right until leaving the grid - tracked the path and checked the number of unique grids visited. Part 2 was running the same walk, but adding one '#' to the grid to see if it makes a cycle. More or less brute forced it except you only need to try to put obstacles in the path followed by part 1.

I did it in python originally and then in Ocaml - for some reason, my ocaml is running slower than my python solution... would like to understand that better lol.

Day 7

We have a bunch of expressions like 156: 15 6 - for part 1 the goal is to figure out if you can get to the number on the left by applying + or * to the numbers on the right, from left to right. Part 2 is the same, but with a concatenation op thrown in there. Honestly couldn't really get going on this one - I went the recursive route first and I think I as almost there but I had some weird bugs that were tough to track down, so I switched to do it iteratively. Part 1 has 2 ops, part 2 has 3, so they have 2^n-1 and 3^n-1 possibilities for the ops in the expression - just checked them all and if it matched the target number, I added it to the count. I bet there's a DP solution to this maybe? Will check out the subreddit later, and revisit this one in Ocaml tomorrow.

EDIT: I realized that I was exiting too early in the recursive soln in stead of or'ing all the subcases together - oh well - would have been a bit faster. I just implemented in ocaml - and IMO it's really nice, honestly should have done this one in ocaml first, I just like the string manipulation in python better

EDIT2: Read about how others solved and implemented the backwards solution - if you start from the right and go left you can trim out way more possibilities and it runs way faster.

Day 8

This was a cool one - a little bit hard to parse the problem description to be honest - but it was finding the two locations (antinodes) that were along the same slope of each pair of specific points in the grid, as long as each "antinode" point was 2 times the distance between one of the points, and 1 times the distance from the other - I just point 1 + 2 * (dr, dc) and point 2 - 2 * (dr, dc) - where (dr, dc) is the vector between the two points. For part to is the same idea, but you need to get the all the valid grid points on the line between them. The trick I think was to get the gcd and then use it to reduce the slope to make sure you step through all the int grid points on the line. Did it in Python to start, will do it in Ocaml "tomorrow" (today).

EDIT: did it in ocaml - this was another pretty tricky one for me to translate.

Day 9

This was a tricky one to get right - for part 1 I read in the data and the moved two pointers along from start and end, swapping whenever there was an empty spot left and a full sport right. Part 2, switched to do it in blocks for each either file or empty spot. For each file from the right, stuck it into the first block that fits it. Not the most optimal - but it works. Really didn't see a nice way to do it in ocaml in a purely fp way, I saw some deque implementations that I might try to copy.

Day 10

The input was a grid of numbers from 0 to 9 - we had to find (part 1) the number of distinct 9's that you could reach from 0 while only moving up in value (and left, right, up, down on the grid). The second part the score for each 0 (starting point) is the number of valid paths that get to a 9, instead of the number of distinct 9's - actually used the same dfs function for both parts, returned the actual paths and then processed them differently for each part. I did it in Python first, and then in Ocaml - I actually really like the ocaml version this time!

Day 11

This was a bunch of 'stones' that updated based on their current number at each 'blink', so 0 turned into 1, etc, based on the given rules. The first part was very straightforward - just run the rules for 25 iterations and count the number of stones. The second, it was impossible to use brute forces because one of the rules resulted in stones splitting into two, so it blew up. The way I actually solved was to track how many of each stone there is at each 'blink' instead of running it for all of the stones. Not before spending a lot of time trying to solve it by caching the results vertically or looking for a cycle in the results (both didn't work for me).

I've done it Python so far, will add in Ocaml. I also found others did have success with using recursion and memoization to cache the results vertically - it's a really interesting solution. The memo table is ( number of blinks left, stone) -> number of stones it turns into after those blinks. Wish I had thought of that but glad I have it for next time!

EDIT: added an ocaml version after getting a better understanding of the recursive solution - used a hashtable for the memo

Day 12

The first part was counting the areas and perimeters of regions in a grid. It was straightforward. The second part was a minor change that asked for the number of sides of the region instead of the perimeter, which was much trickier to get right. I didn't complete it at open this time - had to put it down and go to bed. I was trying to detect edges along both directions and then use that to count the sides, but I ended up just counting the number of corners instead.

Day 13

This was two linear equations that we had to solve - a brute force solution was possible for part 1, you needed to actually solve Ax=b for part 2. I refactored to use the solve for both parts - I actually went for the linear solve right away - but I had an issue with rounding (I was using int() instead of round()) - and the brute-force solution helped me find the problem. Only python for this one and the previous one - I think I'll add ocaml at some point but I've been a little busy that last couple of days - day 12 part 2 took a lot out of me to get working.

EDIT: simpler than I realized - should have written it down, only two eqnx and two unknowns, don't need np or anything fancy, just eliminate one of the unknowns solve and plug back in

Day 14

This was an interesting one! The first part was straightforward to move a bunch of robots on a grid based on their positions and velocities. No real trick except to handle bookkeeping errors, etc. For the second part, we're told that the robots will eventually make a Christmas tree shape and we need to find the first time that happens. I just ran it and tracked the average number of neighbors for each robot at each time step - when it was the highest I've seen so far I printed the grid and the time step and saved it. That turned out to work!

Day 15

This walking around a 'warehouse' grid pushing boxes around. Originally, I did part 1 iteratively (you could only push boxes if there was eventually a free spot behind them) - so I checked all the way to the end in the direction that was relevant - and if was good, swapped the boxes along that dir. Part 2 was trickier, because boxes could be two spaces long - so when going up or down, the box had to move together, eg both spaces had to be free. It was way easier to do it recursively, so to check my understanding I switched to do it that way for part 1 as well. After getting that working, did the same thing for part 2, just needs some extra checking to make sure both spaces can move up / down.

One thing I'm wondering about - there might be a way to do it in 'one pass' like part 1 if you set up the function correctly. The way I set it up, it first first checks if the left part and the right part of the box can move recursively (but doesn't move) and then tries to move them if it can - I had to do it that with the way I set it up because it kept moving just one side otherwise when I didn't want it to.

Also used pattern matching in python like it's a functional language lol

Day 16

Walking around a maze - needed to use dijkstra's algorithm to find the lowest cost path - the trickier part was finding all of the actual lowest cost paths. The idea is pretty straightforward - just keep track of the parent of each cell as you go through the maze, the tricky part was figure out when to overwrite it (if this is the lowest cost path to this node etc) or append to it (if there is a tie). I had a cycle in there for a while that I couldn't quite figure out.

Day 17

Given some custom instructions and reg contents, part 1 was to run the program and enter the values it outputs. Part 2 was wild, had to find a register start val that would make the program output itself. I tried to brute force it but it was not possible, had to write it out and notice the it's doing some ops on 3 bits of the value in the register at a time, so each output is a function of the previous 3 bits, you can solve for the reg value recursively. Did it in python and ocaml this time.

Really hoping for a break in the next few days lol

Day 18

I think we got our break, scared for tomorrow though. This was a 2d grid - trying to get from top left to bottom right of the grid with some obstacles in the way. I used the same dijkstra implementation as a 16 basically - but it was simpler here. We have a list of locations - and in part you add them to grid until you can no longer reach the bottom right (the answer is the first coord results in your path being blocked). I used a binary search on the solution for part 1 to get the answer for part 1.

Day 19

This was determining whether a bunch of strings could be made by a set of given substrings - I don't know why but this felt similar to one of the problems from 2015 for 2025 I think it was nice and easy - I basically got it first shot which never happens. Did it in python and ocaml - uses recursion and caching to make it not super slow - it checks if the str begins with any of the allowed substrs and if it does, calls it self again on the rest of the string.

Day 20

Another grid / maze problem - this time there was only one valid path so it was less of maze. I found the distance of each valid point to the the end and saved it. Then the first part was about finding how many points where you could skip a wall and shortcut the path. The second was the same, but instead of skipping though a single wall block, you can could go up to 20 steps through walls. I took the same part 1 approach but did a dfs from each point on the path to find all the potential paths - and then calculated the gain from each one that ended on a spot on the path

Day 21

This one is a beast. You're pushing buttons to control a robot controlling a robot controlling a robot to push buttons... Definitely feels like I should be able to do it recursively and memo it but no idea how to formulate it. Actually generating even for the sample is out using the naive way... (see below it is possible to generate for pt1, I just had an extra robot layer)

Going to think about it a bit more and peek hints if I get nowhere

UPDATE: I was actually on the right track for part 1 but I had put an extra robot in there, so I was bumping into pt2 scale problem before even solving pt1. In looking for hints, I found that method to generate the paths on the keypad was silly, can go back to taking the difference and just make sure to filter out ones that cross None. Actually thinking that it could just be hardcoded, eg specific shortest ways to get from one key to any other key. I know that pt 2 is based on a caching and memoization now for sure, but still don't 100% see how to formulate it into subproblems, tbd.

UPDATE 2: I understand it now, I had to look at youtube and reddit for hints otherwise I don't think there was anyway I was getting there. Specifically this walkthrough (https://youtu.be/q5I6ZvJmHEo?si=bIkIEbEPYTKAhI63) and this post and comments in that thread (https://www.reddit.com/r/adventofcode/comments/1hja685/2024_day_21_here_are_some_examples_and_hints_for/)

Day 22

This was mostly straightforward - compute a number based on some instructions for part 1 - for part two cached the sequences as the came up and summed after, could definitely be done in fewer passes but going back to work on 21...

Day 23

Had to find cycles of length three and the 'complete subgraphs' or 'cliques' in a graph. Was a pretty straightforward day, but I learned about "Bron-Kerbosch" so that's cool:

Day 24

Part 1 was just like 2015 day 7 - used almost the same code. Currently pretty stuck on part 2. We're supposed to have an adder circuit that can add two binary numbers (In the x and y registers) and put the result in the z registers - it doesn't work but we're given that there are exactly 4 pairs of outputs that need to be swapped to make it work. Generating all the possibilities was too slow, there are a lot. Currently thinking about "fixing" the least significant bit first, and then working up from there? Not sure yet.

UPDATE: checked for each smallest wrong bit, drew out the adder and fixed them one by one. I don't have an algorithmic solution for it yet...

Day 25

Merry Christmas! 🎄