Day 09

Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)

Day 09

My code is awful today

Disgusted cat

The input

The input is pretty simple: We have digits representing a number of blocks. Even digits represent files, with ID 0, 1, 2 etc. Odd digits represent empty spaces.

So 1234 means:

0..111....

I decided to represent the input as a list of (Maybe Int, Int):

In order to get that list, I zip each digits with the list [Just 0, Nothing, Just 1, Nothing …], which I get by interspesing Nothing with the infinite list of Just i, with I starting from 0.

I filter all entries of length 0 as they serve no purpose:

parseInput :: String -> Input
parseInput input = filter ((/= 0) . snd) $ zip blockTypes numbers
    where blockTypes = intersperse Nothing [Just i | i <- [0 .. ]]
          numbers    = map digitToInt . head . lines $ input

Part 1

The problem

I have to try filling empty spaces with the right-most files.

The solution

I start by getting a queue of files that I can use to fill my spaces with:

    where endBlocks = reverse (filter (isJust . fst) input)

Then I simply go through my whole space:

moveBlocks :: Input -> Input
moveBlocks input = go input endBlocks
    where endBlocks = reverse (filter (isJust . fst) input)
          go xs ((Just y, i) : _)
            | x == y = [(Just y, i)]
            | x >  y = []
            where Just x = fst . head $ dropWhile (isNothing . fst) xs
          go ((Nothing, i) : xs) ((Just y, j) : ys)
            | i == j    = (Just y, i) : go xs ys
            | i <  j    = (Just y, i) : go xs ((Just y, j - i) : ys)
            | otherwise = (Just y, j) : go ((Nothing, i - j) : xs) ys
          go (h@(Just _, _) : xs) ys = h : go xs ys

Once I have that, I can compute the checksum by summing all the IDs by their block index.

I compute the indices by stepping by each entry’s length:

computeChecksum :: Input -> Output
computeChecksum = fst .
                  foldl (\(acc, idx) (x, l) -> 
                          (acc + fromMaybe 0 x * sum [idx .. idx + l - 1],
                           idx + l))
                  (0, 0)

partOne :: Input -> Output
partOne = computeChecksum . moveBlocks

Part 2

The problem

Now, instead of filling the left-most empty chunks with the right-most files, I have to move the entire right-most files to the big enough left-most empty chunk.

The solution

In order to do that, I simply go through my space backwards.

moveEntireBlocks :: Input -> Input
moveEntireBlocks = reverse . go  . reverse
    where go [] = []
          go (x@(Nothing, i) : xs) = x : go xs
          go (x@(Just  _, i) : xs)
            | any (canFitFileInBlock x) xs = (Nothing, i) : go xs'
            | otherwise                    = x : go xs
            where xs' = reverse $ placeFile x $ reverse xs
          canFitFileInBlock _ (Just _, _) = False
          canFitFileInBlock (_, i) (Nothing, j) = i <= j 
          placeFile file@(Just v, i) xs
            | i < j  = before ++ (file : (Nothing, j - i) : after)
            | i == j = before ++ (file : after)
            where (before, (_, j) : after) = break (canFitFileInBlock file) xs

partTwo :: Input -> Output
partTwo = computeChecksum . moveEntireBlocks

The end part

Ew, I hate my code. Usually I mainly use a lot of function composition, but today’s solution felt intrasectly iterative, and I needed to write the recursive function to mimick that iterativeness.