DEV Community

Cover image for Trash Compactor
Robert Mion
Robert Mion

Posted on

Trash Compactor

Advent of Code 2025 Day 6

Part 1

Oh what fun! And a preview of Part 2?

This feels like an altogether new challenge. That is, the main part of separating each set of numbers and performing the math.

I also notice that in both the example input and my puzzle input the text alignment of vertical lists differ: some are left-justified, others are right-justified, and some contain numbers that are each the same length.

I have a hunch that Part 2 will require programmatically determining each justification...possibly in order to determine a subset of equations.

Until then, it's time to talk through how I intend to solve Part 1.

Going from four lines to nested arrays

In my head, I want to go from this text string...

123 328  51 64 
 45 64  387 23 
  6 98  215 314
*   +   *   +  
Enter fullscreen mode Exit fullscreen mode

To this data structure:

[
    [[123, 45, 6], '*'],
    [[328, 64, 98], '+'],
    [[51, 387, 215], '*'],
    [[64, 23, 314], '+']
]
Enter fullscreen mode Exit fullscreen mode

Taking this one step at a time...

I can split the string at each newline character:

let input = input.split('\n')
Enter fullscreen mode Exit fullscreen mode

That gets me a list with four strings. The last one is of the operators. They are all a single character, which should make it easy to isolate them:

let operators = input[input.length - 1].replaceAll(' ', '').split('')
Enter fullscreen mode Exit fullscreen mode

I chose to remove all white space and make a list of each character.

As for the other lines, each number and the amount of space between numbers varies.

I want to use a regular expression in a split() to break up the string into a list of numbers with whitespace removed.

I suppose I could also:

  • split() at each character
  • filter() out all space characters

That seems like a fine solution:

input = input.slice(0, -1).map(line => line.split('').filter(i => i !== ' '))
Enter fullscreen mode Exit fullscreen mode

Does it work?

Sure doesn't. Why not?

Because it chops each number up into digits.

Silly me.

New strategy: use regex.

I'll extract each number from the string using a regular expression:

input = input.slice(0, -1).map(line => [...line.matchAll(/\d+/g)].map(el => +el[0]))
Enter fullscreen mode Exit fullscreen mode

It works! I have three lists, each with the numbers in a line.

Now I need to put the numbers in the same index but different lists together.

I'll use a good-'ol-fashioned nested for loop:

let equations = []
for (let i = 0; i < input[0].length; i++) {
    let eq = []
    for (let j = 0; j < input.length; j++) {
        eq.push(input[j][i])
    }
    equations.push([eq, operators[i]])
}
Enter fullscreen mode Exit fullscreen mode

What this should do is:

  • Create an empty array
  • Iterate as many times as there are numbers in a line (the same for each line)
  • Create an empty array
  • Iterate as many times as there are lines - minus the last line which has been removed by now
  • Add the number at the current index to the empty array
  • Finally, add the empty array and the corresponding operator as a 2-item list to the other empty array

Does this work?

It does!

[
  [ [ 123, 45, 6 ], '*' ],
  [ [ 328, 64, 98 ], '+' ],
  [ [ 51, 387, 215 ], '*' ],
  [ [ 64, 23, 314 ], '+' ]
]
Enter fullscreen mode Exit fullscreen mode

Woohoo!

Next up: Math!

Processing each equation and testing

I intend to use a reduce() and eval():

  • The former as a convenient syntax to arrive at a single number
  • The latter as a convenient method of performing math without having to map each string character to its programmatic operator

After a bit of debugging, my working algorithm looks like this:

let part1 = equations.reduce((answer, eq) => {
    return answer += eq[0].reduce((total, num) => {
        return eval(`${total} ${eq[1]} ${num}`) 
    })
}, 0)
Enter fullscreen mode Exit fullscreen mode

It generates the correct answer when runnning the example input.

Will it do the same when running my input?

...

It will! It did!

Woohoo!!

I'm glad Part 1 didn't have any hidden tricks in the input.

Now, will Part 2 make use of the varying text alignment of each equation?

I can't wait to find out.

Part 2

I knew it! Also, I had no clue.

I was right about the text alignment coming into play!

And yet the way it comes into play goes beyond what I imagined...

...into much cooler challenge territory.

This is gonna be a blast to solve...I hope.

Finding a new strategy for processing each column

The convenient thing is the operators are still in the last line.

The only difference is I'll need to process them in reverse chronological order.

I think I can carry forward my code that extracted the last line of input:

input = input.split('\n')
let operators = input[input.length - 1].replaceAll(' ', '').split('').reverse()
Enter fullscreen mode Exit fullscreen mode

Visually, that statement should go like this:

input[input.length - 1]
*   +   *   +  

.replaceAll(' ', '')
*+*+

.split('')
['*','+','*','+']

.reverse()
['+','*','+','*']
Enter fullscreen mode Exit fullscreen mode

Now that the operators are removed, my input string looks like this:

123 328  51 64 
 45 64  387 23 
  6 98  215 314
Enter fullscreen mode Exit fullscreen mode

I need to programmatically perform the equivalent of a 90-degree counterclockwise turn to get the numbers properly oriented, like this:

  4
431
623

175
581
 32

8
248
369

356
24
1
Enter fullscreen mode Exit fullscreen mode

I plan to write another nested for loop to iterate backwards through the same index of each line and build a new array of numbers.

First, I need to walk through an example to understand how I will access each index.

To get the first number, 4, I need to access these indices:

input[0][input[0].length - 1]
input[1][input[1].length - 1]
input[2][input[2].length - 1]
Enter fullscreen mode Exit fullscreen mode

I should be able to refer to my local loop variables instead of re-typing input[N].length - 1.

Now for the algorithm:

let equations = []
for (let i = input[0].length - 1; i >= 0; i--) {
    let eq = []
    for (let j = 0; j < input.length; j++) {
        eq.push(input[j][i])
    }
}
Enter fullscreen mode Exit fullscreen mode

When I run it wth a logging statement, I see what I expected:

[ ' ', ' ', '4' ]
[ '4', '3', '1' ]
[ '6', '2', '3' ]
[ ' ', ' ', ' ' ]
// ...and so on
Enter fullscreen mode Exit fullscreen mode

Next, I need to convert each list of strings into either a number or an indicator to start a new set of numbers.

So, in the case of the above output, turn those four list of strings into:

[ [4, 431, 623], ... ]
Enter fullscreen mode Exit fullscreen mode

I'm confident I can concatenate the strings and convert to a number.

Except for the list of all space characters, as that may not be a number.

So I will need a condition to handle that case.

Here's my algorithm:

let equations = [[]]
for (let i = input[0].length - 1; i >= 0; i--) {
    let eq = []
    for (let j = 0; j < input.length; j++) {
        eq.push(input[j][i])
    }
    let num = +eq.join('').trim()
    if (num == 0) {
        equations.push([])
    } else {
        equations[equations.length - 1].push(num)
    }
}
Enter fullscreen mode Exit fullscreen mode
  • I added a nested empty list to equations
  • The second half of the nested for loop is new
  • num is the list of strings joined, with white space removed, then coerced to a number
  • The condition adds a new nested list when it sees a 0
  • Otherwise it inserts the number at the end of the last list

Thankfully, I see the expected result:

[ [ 4, 431, 623 ], [ 175, 581, 32 ], [ 8, 248, 369 ], [ 356, 24, 1 ] ]
Enter fullscreen mode Exit fullscreen mode

Now I need to reduce each sub-list by applying the correct operator to each, just like in Part 1.

That code with some minor tweaks to work with my updated data structure looks like:

let part2 = equations.reduce((answer, eq, idx) => {
    return answer += eq.reduce((total, num) => {
        return eval(`${total} ${operators[idx]} ${num}`) 
    })
}, 0)
Enter fullscreen mode Exit fullscreen mode

I think that's it, since, when I run it, I see the expected answer for the example input!

But will it generate the correct answer for my puzzle input?

...

It does!

Woohoo!!!

That was such a fun 2-part challenge!

I just knew the varyied text alignment would come into play.

Top comments (0)