Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
It is already time for 2D array manipulation in Haskell! 😿
import Data.Char
import Data.List
import Data.Matrix (Matrix, fromLists, (!), nrows, ncols)
import System.Environment
data Number = Number { value :: Int, row :: Int, span :: (Int, Int) } deriving (Show)
-- Grid, Numbers, Potential Gears
type Input = (Matrix Char, [Number], [(Int, Int)])
type Output = Int
parseInput :: String -> Input
parseInput input = (grid, numbers, gears)
where inLines = lines input
grid = fromLists inLines
-- Getting the list of numbers
numCol = map (zip [1 .. ]) inLines
grouped = map (groupBy (\a b -> isDigit (snd a) == isDigit (snd b))) numCol
cleaned = map (map (\l -> ((fst $ head l, fst $ last l), map snd l))) grouped
numRow = zip [1 .. ] $ map (filter (\(_, n) -> all isDigit n)) cleaned
numbers = concatMap (\(r, row) -> map (\(s, n) -> Number (read n) r s) row) numRow
-- Getting the list of gears
gearCols = map (filter ((== '*') . snd )) numCol
gearRows = zip [1 .. ] gearCols
gears = concatMap (\(r, row) -> map (\(c, _) -> (r, c)) row) gearRows
partOne :: Input -> Output
partOne (grid, numbers, _) = sum . map (value) . filter isAdjacent $ numbers
where maxRow = nrows grid
maxCol = ncols grid
getNeighbours (Number _ r (c1, c2)) = map (grid !) [(i, j) | i <- [r - 1 .. r + 1], j <- [c1 - 1 .. c2 + 1], 0 < i && i <= maxRow, 0 < j && j <= maxCol]
isAdjacent = any (\c -> not (isDigit c) && c /= '.') . getNeighbours
partTwo :: Input -> Output
partTwo (grid, numbers, gears) = sum . map (product . map value) . filter ((== 2) . length) . map getNeighbours $ gears
where maxRow = nrows grid
maxCol = ncols grid
getNeighboursPos (r, c) = [(i, j) | i <- [r - 1 .. r + 1], j <- [c - 1 .. c + 1], 0 < i && i <= maxRow, 0 < j && j <= maxCol]
getNeighbours g = filter (\(Number _ r (c1, c2)) -> any (`elem` neighbours) [(r, c) | c <- [c1 .. c2]]) numbers where neighbours = getNeighboursPos g
There is a lot to unpack here so let’s start by the input:
-- Grid, Numbers, Potential Gears
type Input = (Matrix Char, [Number], [(Int, Int)])
There are three interesting parts here: the actual grid (basically the input, almost raw, just wrapped inside a Data.Matrix), the list of numbers and the list of potential grids represented by their coordinates.
Numbers are represented with a data structure containing their value, their row, and their span (first column and last column)
Parsing the input is hellish:
parseInput :: String -> Input
parseInput input = (grid, numbers, gears)
where inLines = lines input
grid = fromLists inLines
-- Getting the list of numbers
numCol = map (zip [1 .. ]) inLines
grouped = map (groupBy (\a b -> isDigit (snd a) == isDigit (snd b))) numCol
cleaned = map (map (\l -> ((fst $ head l, fst $ last l), map snd l))) grouped
numRow = zip [1 .. ] $ map (filter (\(_, n) -> all isDigit n)) cleaned
numbers = concatMap (\(r, row) -> map (\(s, n) -> Number (read n) r s) row) numRow
-- Getting the list of gears
gearCols = map (filter ((== '*') . snd )) numCol
gearRows = zip [1 .. ] gearCols
gears = concatMap (\(r, row) -> map (\(c, _) -> (r, c)) row) gearRows
There are three main parts:
where inLines = lines input
grid = fromLists inLines
This is the easy part: get the lines from the input, and transform it into a matrix
Now for the hard part:
-- Getting the list of numbers
numCol = map (zip [1 .. ]) inLines
grouped = map (groupBy (\a b -> isDigit (snd a) == isDigit (snd b))) numCol
cleaned = map (map (\l -> ((fst $ head l, fst $ last l), map snd l))) grouped
numRow = zip [1 .. ] $ map (filter (\(_, n) -> all isDigit n)) cleaned
numbers = concatMap (\(r, row) -> map (\(s, n) -> Number (read n) r s) row) numRow
I start by labelling each column in each row, then I group every number digit together (so I get the numbers whole in the form [(col1, digit 1), (col2, digit2) …]).
After that, I clean things around by taking only the first and last column for each digit, and putting the digits together in a list: ((col1, colLast), [digit1, digit2, …])
Now that I have that, I only keep the actual numbers by filtering on the digits to make sure they’re ACTUAL digits, and I now label the rows.
Finally, for each row I transform every (span, digits) into a number data structure, and I flatten my grid into a single list.
Now the second hardest part:
-- Getting the list of gears
gearCols = map (filter ((== '*') . snd )) numCol
gearRows = zip [1 .. ] gearCols
gears = concatMap (\(r, row) -> map (\(c, _) -> (r, c)) row) gearRows
First I keep the * from the labeled column list, next I label the rows, and finally I transform these into coordinates.
This was the hard part (the list of gears was done after I did part one btw)
Now for the puzzles themselves:
partOne :: Input -> Output
partOne (grid, numbers, _) = sum . map (value) . filter isAdjacent $ numbers
where maxRow = nrows grid
maxCol = ncols grid
getNeighbours (Number _ r (c1, c2)) = map (grid !) [(i, j) | i <- [r - 1 .. r + 1], j <- [c1 - 1 .. c2 + 1], 0 < i && i <= maxRow, 0 < j && j <= maxCol]
isAdjacent = any (\c -> not (isDigit c) && c /= '.') . getNeighbours
I use a function getNeighbours that, for a given number, generates the list of possible neighbour coordinates and maps this list to the values in the grid. Then I simply need to check if any of these values is a symbol (ie not . nor number) to know that this number is adjacent to one.
I filter the numbers that are adjacent to a symbol, get their values and sum them together!
Part two is similar:
partTwo :: Input -> Output
partTwo (grid, numbers, gears) = sum . map (product . map value) . filter ((== 2) . length) . map getNeighbours $ gears
where maxRow = nrows grid
maxCol = ncols grid
getNeighboursPos (r, c) = [(i, j) | i <- [r - 1 .. r + 1], j <- [c - 1 .. c + 1], 0 < i && i <= maxRow, 0 < j && j <= maxCol]
getNeighbours g = filter (\(Number _ r (c1, c2)) -> any (`elem` neighbours) [(r, c) | c <- [c1 .. c2]]) numbers where neighbours = getNeighboursPos g
This time, I first generate the list of neighbouring coordinates, and then I filter the number list by checking if any matches these coordinates.
Then for each gear I keep those with exactly two neighbours, I get the value of these neighbours and multiply them, and I sum all the results.
I am not happy with that 😿