Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
Welcome to “let’s just print everything and look-up manually”’s day!
Each line represents a robot with a position and a velocity.
I first defined a data type representing that robot:
data Robot = Robot { position :: (Int, Int), velocity :: (Int, Int) } deriving (Show)
Then, for each line I extract the position and velocity part of the line, and I convert them to tuple of Int:
parseInput :: String -> Input
parseInput = map parseRobot . lines
where parseRobot :: String -> Robot
parseRobot line = Robot pos vel
where [_ : p : v : _] = line =~ "p=([0-9]+,[0-9]+) v=(-?[0-9]+,-?[0-9]+)" :: [[String]]
pos = read ("(" ++ p ++ ")")
vel = read ("(" ++ v ++ ")")
I want to know the robots’ positions after 100 seconds.
I first define a function that computes a robot’s position after a given number of seconds.
A robot’s position is computed with the following formulas:
size :: (Int, Int)
size = (101, 103)
move :: Int -> Robot -> Robot
move seconds r = r'
where (width, height) = size
(x, y) = position r
(dx, dy) = both (* seconds) $ velocity r
pos' = ((x + dx) `mod` width, (y + dy) `mod` height)
r' = r { position = pos' }
Now, I need to be able to sort a robot by its quadrant. That is simply comparing its coordinates to the middle row and column:
isInQuadrant :: Robot -> Int
isInQuadrant (Robot (x, y) _)
| x < midX && y < midY = 1
| x > midX && y < midY = 2
| x < midX && y > midY = 3
| x > midX && y > midY = 4
| otherwise = 0
where (midX, midY) = both (`div` 2) size
Quadrant 0 corresponds to the middle line or column.
Now I can compute the safety factor by simply getting the number of robots in each quadrant and multiplying these numbers together:
computeSafetyFactor :: Int -> Input -> Output
computeSafetyFactor seconds = product .
map length .
group . sort .
filter (/= 0) .
map (isInQuadrant . move seconds)
partOne :: Input -> Output
partOne = computeSafetyFactor 100
I have to find when these robots form a christmas tree.
I have no idea what that christmas tree looks like, nor how big it is.
I start by finding the string representation of the grid after a given number of seconds.
I represent robots with “#” and empty tiles with “ “.
getGridAfter :: Int -> Input -> [String]
getGridAfter seconds rs = [ [ getRepresentingChar x y | x <- xs ] | y <- ys ]
where (width, height) = size
rs' = map (move seconds) rs
xs = [0 .. width - 1]
ys = [0 .. height - 1]
getRepresentingChar x y | any (\r -> position r == (x, y)) rs' = '#'
| otherwise = ' '
Then, I print a lot of grids…
partTwo :: Input -> IO ()
partTwo input = for_ [0 .. fst size * snd size] $ \i -> do
print i
mapM putStrLn . getGridAfter i $ input
And now, I just launch that and I look manually.
In order to be more efficient (or less inefficient), I use a grep
to find lines with multiple “#” (as it will probably give me a better chance of finding the tree):
./Day two input | fgrep -B 101 "#####" > result
Read that with less
, and enjoy your horrible moment
I don’t like puzzles when I don’t know what I’m supposed to look for :c