Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
Repeat after me:
Manipulating 2D grids in Haskell isn’t fun!
Also, remember that I’m not going for the most optimised solution, but for the one that doesn’t require a lot of code!
The input is a simple grid of characters, so we just get each line:
MMMSXXMASM
MSAMXMSMSA
AMXSXMAAMM
MSAMASMSMX
XMASAMXAMM
XXAMMXXAMA
SMSMSASXSS
SAXAMASAAA
MAMMMXMMMM
MXMXAXMASX
parseInput :: String -> Input
parseInput = lines
We need to search inside the grid for any “XMAS” horizontally, vertically and diagonally.
Searching horizontally is pretty easy:
We can simply use a regex search on “XMAS”!
However, we also need to find “SMAX”, and using the regex library to search “XMAS|SMAX” won’t work for the case “XMASMAX” as the S is going to be taken by “XMAS” (Same with SMAXMAS). (Maybe there is a parameter for that, but I didn’t find it :x)
One simple thing we can do to solve that is simply reversing the string before searching a second time!
search :: String -> [String]
search l = concat (l =~ "XMAS" :: [[String]])
searchHorizontal l = length $ concatMap (search . reverse) l ++ concatMap search l
Now that we can find all horizontal instances of “XMAS”, searching for all vertical instances is easy: simply rotate (transpose actually) the grid!:
MMMSXXMASM
MSAMXMSMSA
AMXSXMAAMM
MSAMASMSMX
XMASAMXAMM
XXAMMXXAMA
SMSMSASXSS
SAXAMASAAA
MAMMMXMMMM
MXMXAXMASX
->
MMAMXXSSMM
MSMSMXMAAX
MAXAAASXMM
SMSMSMMAMX
XXXAAMSMMA
XMMSMXAAXX
MSAMXXSSMM
AMASAAXAMA
SSMMMMSAMS
MAMXMASAMX
Now we simply have to search horizontally for all XMAS!
searchVertical = searchHorizontal . transpose
Finally, the hardest part: counting diagonally.
In order to do that, I start by getting all subsets of 4 consecutive lines.
In order to do that, I make a chunksOf function that gets all subsets of n consecutives lines:
-- Not the same as Data.List.Split's chunksOf
-- This gives all the complete subsequences of length n
-- That is like iterating over take (length l - x) $ drop x l
chunksOf :: Int -> [a] -> [[a]]
chunksOf n = concatMap (filter ((== n) . length) . inits) . tails
For example, the second subset is the following:
MMMSXXMASM
MSAMXMSMSA
AMXSXMAAMM
MSAMASMSMX
XMASAMXAMM
XXAMMXXAMA
SMSMSASXSS
SAXAMASAAA
MAMMMXMMMM
MXMXAXMASX
->
MSAMXMSMSA
AMXSXMAAMM
MSAMASMSMX
XMASAMXAMM
Then, for each subset, I shift each line by it’s index.
For example our last result will get shifted like this:
MSAMXMSMSA
MXSXMAAMM
MASMSMX
AMXAMM
Now, if there was “XMAS” written diagonally, the letters would all fall on the same column!
In other terms, we now simply need to count vertically!
searchDiagonal l = countInDiagonal l + countInDiagonal (reverse l)
where shift l = [drop i x | (i, x) <- zip [0 .. ] l]
getDiagonals = map shift . chunksOf 4
countInDiagonal = sum . map searchVertical . getDiagonals
If we put all that together:
partOne :: Input -> Output
partOne input = searchHorizontal input + searchVertical input + searchDiagonal input
where search :: String -> [String]
search l = concat (l =~ "XMAS" :: [[String]])
searchHorizontal l = length $ concatMap (search . reverse) l ++ concatMap search l
searchVertical = searchHorizontal . transpose
searchDiagonal l = countInDiagonal l + countInDiagonal (reverse l)
where shift l = [drop i x | (i, x) <- zip [0 .. ] l]
getDiagonals = map shift . chunksOf 4
countInDiagonal = sum . map searchVertical . getDiagonals
And it’s actually not too slow OwO
sh$ time ./Day_04 one input
2549
./Day_04 one input 0.04s user 0.01s system 11% cpu 0.473 total
This part was actually simpler in my opinion :3c
Now, we don’t actually want to find “XMAS” in the grid! We actually want to find crosses that spell “MAS” twice!
For example:
M.S
.A.
M.A
We can simply look through all 3x3 chunks in our grid, and check if it is a ‘MAS’ cross.
There are actually only 4 valid ‘MAS’ crosses:
M.S S.M S.S M.M
.A. .A. .A. .A.
M.S S.M M.M S.S
So I harcoded them in a ‘isCross’ function:
isCross ['M', _, 'S'] [_, 'A', _] ['M', _, 'S'] = True
isCross ['S', _, 'M'] [_, 'A', _] ['S', _, 'M'] = True
isCross ['M', _, 'M'] [_, 'A', _] ['S', _, 'S'] = True
isCross ['S', _, 'S'] [_, 'A', _] ['M', _, 'M'] = True
isCross _ _ _ = False
Now, I get all possible chunks of 3 lines, and for each line I get all the chunks of 3 columns.
I zip the column-chunks of each line to get all 3x3 chunks, and I apply isCross on all of these chunks.
Now I filter the ones that are crosses, and I count their length:
partTwo :: Input -> Output
partTwo = length . filter id . concatMap (zipWith3' isCross . map (chunksOf 3)) . chunksOf 3
where zipWith3' f [a, b, c] = zipWith3 f a b c
isCross ['M', _, 'S'] [_, 'A', _] ['M', _, 'S'] = True
isCross ['S', _, 'M'] [_, 'A', _] ['S', _, 'M'] = True
isCross ['M', _, 'M'] [_, 'A', _] ['S', _, 'S'] = True
isCross ['S', _, 'S'] [_, 'A', _] ['M', _, 'M'] = True
isCross _ _ _ = False
This still runs decently fast:
sh$ time ./Day_04 two input
2003
./Day_04 two input 1.04s user 0.03s system 97% cpu 1.095 total
Of course, I could’ve used a Data.Matrix or some Data.Array, as they have fast index-access functions.
And of course I could’ve then written a function that iterates through each tile and look for ‘XMAS’ there in all 8 directions.
But that didn’t feel right :3