Skip to content

cr0t/advent-of-code-2024

Repository files navigation

Advent of Code 2024

Use Livebook server to run these .livemd files.

$ livebook server
[Livebook] Application running at http://localhost:8080/?token=***

Open the given link and then .livemd file you want.

Input can be found in the similarly-named files.

Solutions' Notes

Day 1: Historian Hysteria

  • Input: Split all the input by whitespace (this gives a list of ALL the given numbers), chunk the list by 2, and transpose it right away to get two lists (one for the 1st column, another for the 2nd column).
  • Part 1: Sort both lists, zip together (to get both {left, right} numbers as one item), find the difference, and sum it up.
  • Part 2: Convert the right list into a Map structure with keys of how many times each number appears in the list (frequencies); then take each item from the left list, find the number of occurrences in the right (using the converted data structure), and sum it up.

Day 2: Red-Nosed Reports

  • Input: Every line is a single piece of work; split it line-by-line, then split each line into items and convert them to integers.
  • Part 1: Express what safe?/1 means for a line of numbers (with an “early exit” based on Enum.reduce_while/3)
  • Part 2: Re-use the same safe?/1 from the first part, but make all possible inputs one step before and select those that are safe.

Day 3: Mull It Over

  • Input: Use Regex to scan the input for the instructions and arguments (they all have a well-defined format).
  • Part 1: Reduce the list of instructions, considering only mul(x, y) ones.
  • Part 2: The similar reduce, but with more rules (expressed as multiple clauses of the reducer function).

Day 4: Ceres Search

  • Input: Convert the input to a map with keys as {x, y} coordinates pointing to values (what's on the given field).
  • Part 1: Find all the starting points for the X character, then (using starting points) build lists of coordinates for possible positions of XMAS letters on the board; using these coordinate, get real values from the map, and compare if we get an 'X', 'M', 'A', 'S' then (if so, add 1 to the hits sum).
  • Part 2: Similar to Part 1, but build X-shaped coordinates for each 'A' starting point (as the middle one), collect letters and compare that both diagonals are 'MAS' (or 'SAM'), and count only those that fully match the pattern.

Visually, it might be represented as sitting next to a board, and applying some form to it (XMAS, or X-shaped MAS) to each starting point and seeing what we get. Like playing in a sandbox and “baking” stuff.

Day 5: Print Queue

  • Input: Input has two sections: rules and updates; rules are better to save as Map, while updates are just a list of lists of integers; we also define a couple of helpers — correct/2 and sum_middles/1 which we use later.
  • Part 1: Filter all correct items, sum their middles…
  • Part 2: Reject all correct items and fix incorrect ones; to fix them, we calculate a “weight” of each item in the list; we take the item's rules, and compare to siblings — more siblings in the rules, more weight; later we just order by weight; and, eventually, sum the middles of the fixed updates.

Day 6: Guard Gallivant

  • Input: It's a simple map, convert it to a Map with keys-coordinates pointing to actual value of the field it represents; in addition max_x and max_y included; we also define helpers to “move” into some direction from a stating position and detecting if we hit an obstacle in the process, or went into edge — we return visited coordinates, including some metadata (where to turn next, for example).
  • Part 1: Find the starting position, feed it to the track/3 function that collects all the visited coordinates until we approach an “edge,” then we flatten, find unique fields, and count them.
  • Part 2: Use the original path to put random obstacles on (not sure why it works, but it is!). It works ~3x times faster in comparison to check EVERY “.” field for possible loops); then take every possible new obstacle and check if we in a loop or not; for that, we have changed the original data structure to check if we are in the same coordinate and same direction easier (there is a Map with keys-directions and values — list of visited previously fields having the same direction).

Day 7: Bridge Repair

  • Input: Regex helps to find all the numbers in the lines; due to format, the very first number in each “equation” is its result, the rest are operands; in addition to parsing, we also place combinations generator which takes a list of possible values and an expecting length and generates combinations of that length with all positions of values.
  • Part 1: Filter equations with the function that (1) generates all possible combinations of operators, (2) “inserts” them between the operands, (3) does the equation, and (4) compares to the result; then find the sum of results.
  • Part 2: Do basically everything as in the 1st part, but (due to the increase in possible combinations) parallelize it with Task.async_stream; also use Stream.map on combinations to reduce the number of eventual computations.

Day 8: Resonant Collinearity

  • Input: Input can be represented using Map (with max_x, and max_y).
  • Part 1: Find all antennas' positions and group them by their “frequency” (label); find all the unique pairs of these antennas and then calculate list of antinodes' coordinates (using Pythagorean theorem we can determine deltas of x and y, and find opposite points on the grid; points' coordinates depend on the mutual position of first and second antennas); we filter only those coordinates that are in the map; moreover, due to antinodes can take the same coordinate, we have to find unique only, and count them.
  • Part 2: We do similar thing as in the 1st part, but now we find not just 2 antinode coordinates for each antenna's pair; we find ALL the antinodes in each direction, starting from each antenna's point and until we reach the edge of the map; the antennas coordinates are also added to the list of antinodes; the rest is similar — uniq, count.

Day 9: Disk Fragmenter

  • Input: It's a very long line of digits, which we must split into single ones, convert to integers, and then chunk every 2 (we can add 0 to the last if needed). We also give an index to each of the pair, because later it will become a file ID.
  • Part 1: To simplify data lookups, we make a list of file blocks (and reverse it). Then we go over each block in the disk map, and try to fit a piece of a file from the reversed blocks list (this helps us to get files' blocks from the farthest side easily). We do so until out (reversed) stash is empty or we processed all the data blocks from the original disk map (excluding trailing empty space). Then calculate checksum.
  • Part 2: It's a trickier version of first, because we have to find a proper hole to move file as a whole. We made a files_meta map with some extra info (for quicker data retrieval). Though, we store the disk map as a list of two-element tuples (with file_id and index of that block - we will need this data at the checksum step). We go over the files (in the descending order for file ID), try to find a proper hole for it and move if hole is found; if hole's not found, we check next file, and so on. When we find a hole, we cut the disk maps list into a few pieces (like in a film editing process), generate new file and hole blocks, and glue them all again. At the end, checksum.

Day 10: Hoof It

  • Input: As the first step, we have to convert the input into a Map with {x, y} coordinates as keys, and height levels as values. Later, we must convert this map into a graph (which is also presented as a Map) with keys as {x, y} points and possible next moves (which are dependent on the neighbor's level) from this position.
  • Part 1: Solved with a slightly modified BFS (Breadth-First Search) algorithm. The modification lets us know whether the target is accessible or not; we are not interested in the path itself.
  • Part 2: Solved with DFS (Depth-First Search) which enables us to find ALL possible paths in a graph between two given nodes.

Day 11: Plutonian Pebbles

  • Input: Simply a list of integer numbers split by whitespace.
  • Part 1: This part can be solved in a brute-force way: by reducing a linked list of the stones on each step, just apply the rules to each stone in the list.
  • Part 2: This part cannot be solved in a brute-force manner — just too many items in the list… The trick is in the data structure: instead of a list of stones, we shall store them as maps. Basically, a statistic of how many of each stone is there at any moment. Here is an example: 1036288 7 2 20 24 4048 1 4048 8096 28 67 60 32 shall be represented as %{1 => 1, 2 => 1, 7 => 1, 20 => 1, 24 => 1, 28 => 1, 32 => 1, 60 => 1, 67 => 1, 4048 => 2, 8096 => 1, 1036288 => 1}. At each step (blink), we apply rules to the map's keys and then convert it to a new statistical map.

Day 12: Garden Groups

  • Tags: map, graph, BFS, neighbors, perimeter, think-outside-of-the-box
  • Input: Read it as a map, but convert to the graph (of connected nodes by neighboring plant types) right away, as we will use BFS to locate the regions of the plants on this map/graph.
  • Part 1: Find all the regions (they will be presented as MapSets) and calculate their perimeters (each item is 4 minus the number of neighbors).
  • Part 2: The trick is in "scanning" the region using vertical and horizontal lines and collecting metadata about the plants, then we can use it to calculate the number of consecutive sides. See more details in the comments in the code.