DEV Community

Cover image for Rope Bridge
Robert Mion
Robert Mion

Posted on

Rope Bridge

Advent of Code 2022 Day 9

Part 1

  1. Manhattan Distance for the win?
  2. Replaying the example to outline my algorithm
  3. My algorithm in JavaScript

Manhattan Distance for the win?

  • I closely studied each diagram
  • I noticed that T only moves diagonally when H is a Manhattan Distance of 3 away from T
  • At that time, T moves such that the Manhattan Distance becomes 1, occupying the last spot H was in
  • In other cases, T either doesn't move, or moves into the last spot occupied by H to maintain a Manhattan Distance of 0 or 1 (same row or column) or 2 (diagonal)

How might I account for these movement rules, without letting any conditionals get way out of control?

It seems that is my first challenge, given how I anticipate writing an algorithm

Replaying the example to outline my algorithm

Initial State:

......
......
......
......
H.....  (H covers T, s)
Enter fullscreen mode Exit fullscreen mode

My initial variables:

H = [ '0,0' ]
T = [ '0,0' ]

directions = {
  'U': [-1,0],
  'R': [0,1],
  'D': [1,0],
  'L': [0,-1]
}
Enter fullscreen mode Exit fullscreen mode

R 4 means execute four movements to the right.

Do 4 times
  Move H to the right
  Determine Manhattan Distance between H and T
  If it is < 2
    Do nothing
  Else if == 2
    If same row or column
      Move T to the last spot occupied by H
    Else - diagonally adjacent
      Do nothing
  Else
    Move T to the last spot occupied by H
Enter fullscreen mode Exit fullscreen mode

I can abstract that for any direction and movement amounts.

DIR AMT

Do AMT times
  Move H to the DIR
  Determine Manhattan Distance between H and T
  If it is < 2
    Do nothing
  Else if == 2
    If same row or column
      Move T to the last spot occupied by H
    Else - diagonally adjacent
      Do nothing
  Else
    Move T to the last spot occupied by H
Enter fullscreen mode Exit fullscreen mode

Sub-routines:

  • Move H to the DIR
  • Determine Manhattan Distance
  • Move T to the last spot occupied by H

Move H to the DIR

  • I made H an array containing stringified coordinates
  • Each movement adds an item at the end of the array
  • The item is the result of adding each amount in the directions object to the corresponding coordinate of the last value in H

Determine Manhattan Distance

  • Calculate the absolute values of the difference of each of H and Ts coordinates
  • Calculate the sum of both values

Move T to the last spot occupied by H

  • T is also an array and works like H
  • Each movement of T adds an item at the end of the array
  • That item is a string copy of the the last item in H

Back to the example.

R 4 looks like this:

TH...

sTH..

s.TH.

s..TH
Enter fullscreen mode Exit fullscreen mode

My H and T arrays would look like this:

H    T    MD
0,0  0,0  0
0,1  0,0  1
0,2  0,0  2   Same row: Move T to 0,1
0,3  0,1  2   Same row: Move T to 0,2
0,4  0,2  2   Same row: Move T to 0,3
Enter fullscreen mode Exit fullscreen mode

U 4 looks like this for my arrays:

 H     T    MD
 0,4   0,3  1
-1,4   0,3  2   Diagonal: Don't move
-2,4   0,3  3   Move T to -1,4
-3,4  -1,4  2   Same column: Move T to -2,4
-4,4  -2,4  2   Same column: Move T to -3,4
Enter fullscreen mode Exit fullscreen mode

I feel confident enough that I have identified enough cases in my planned algorithm to now attempt writing it.

This should be a doozy!

My algorithm in JavaScript

  • Writing the pseudocode really helped!
  • It works exactly as I designed it!
  • With the exception of storing nested arrays, then stringifying all of T's coordinates at the very end to generate the list of unique spots visited
let H = [[0,0]]
let T = [[0,0]]
let moves = {
  'U': [-1,0],
  'R': [0,1],
  'D': [1,0],
  'L': [0,-1]
}
let cmds = input.split('\n').map(el => {
  let [dir, amt] = el.split(' ')
  return [dir, +amt]
})
cmds.forEach(([dir, amt] = cmd) => {
  for (let i = 0; i < amt; i++) {
    H.push([
      H[H.length - 1][0] + moves[dir][0],
      H[H.length - 1][1] + moves[dir][1]
    ])
    let MD = (
      Math.abs(H[H.length - 1][0] - T[T.length - 1][0]) +
      Math.abs(H[H.length - 1][1] - T[T.length - 1][1])
    )
    switch (MD) {
      case 0:
      case 1:
        break;
      case 2:
        if (
          H[H.length - 1][0] == T[T.length - 1][0] ||
          H[H.length - 1][1] == T[T.length - 1][1]
        ) {
          T.push(H[H.length - 2])
        }
        break;
      default:
        T.push(H[H.length - 2])
        break;
    }
  }
})
return new Set(T.map(el => el.join('|'))).size
Enter fullscreen mode Exit fullscreen mode

Part 2

Not crossing this bridge

Why not?

In the updated example diagram, there is this before-and-after:

......
......
......
....H.
4321..

......
......
....H.
.4321.
5.....
Enter fullscreen mode Exit fullscreen mode
  • I know how to programmatically describe what makes 1 move too its new spot
  • I understand why 2, 3 and 4 end up where they do
  • But I fail to think of a way to programmatically ensure they would move like that

As the diagrams continue to depict how all 10 knots move in the original example, I feel further puzzled about each knots movement...especially in how I would recreate such movements algorithmically.

Thus, sadly, Part 2 will go unsolved for me.

I did it!

  • I solved Part 1!
  • Using Manhattan Distance and arrays in lieu of a litany of nested conditionals!

I'm bummed about this being my second time that I only earned one star on a Day 9 - in all other years I earned two stars.

But, as with any Day, I'm glad I solved Part 1!

Top comments (0)