Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
Today was a fun day! Not too hard, and quite fun!
In order to represent bricks, I use the following data structure:
data Brick = Brick { minZ :: Int, maxZ :: Int, cubes :: S.Set (Int, Int, Int), supporting :: S.Set Int} deriving (Eq, Ord)
Here is what each field means:
minZ and maxZ are my first field in my structure, simply because I want to compare bricks by their z coordinate first and foremost.
This will be very useful later!
For once, instead of making a function for parsing the input, I simply make Brick an instance of Read (allowing me to simply call the read function on a line of my input file):
instance Read Brick where
readsPrec _ s = [(Brick minZ maxZ cubes S.empty, "")]
where (edge1, '~':edge2) = break (=='~') . head . words $ s
((x1, y1, z1), (x2, y2, z2)) = both (read . ('(' :) . (++ ")")) (edge1, edge2)
minZ = min z1 z2
maxZ = max z1 z2
cubes = S.fromList [(x, y, z) | x <- [min x1 x2 .. max x1 x2]
, y <- [min y1 y2 .. max y1 y2]
, z <- [min z1 z2 .. max z1 z2]]
The way this works is:
To have some fun, I’ve also made Brick an instance of Show:
-- This one is just for fun (and easier debugging in case there are like 1000 cubes in a brick)
instance Show Brick where
show (Brick _ _ c s) = firstCube ++ "~" ++ lastCube ++
" | Number of cubes = " ++ show (S.size c) ++
" | Supported by = " ++ show s
where firstCube = init . tail . show $ S.findMin c
lastCube = init . tail . show $ S.findMax c
This allows me to pretty print my input in a format that is similar to the original input format.
Interesting to note here is that (read . show) brick == id brick (because I split by spaces first in my readsPrec!) :D
Our list of initial state bricks is actually quite uninteresting for us. What we really want to work with is the list of bricks AFTER they’ve fallen down. This is the meatiest part of my solution, so here we go!
The basic idea is the following:
While this idea works, it is not the fastest! One simple speedup that can be done (and that I did) is that:
Put that into code:
fallDown :: [Brick] -> M.Map Int Brick
fallDown = foldl' go M.empty
where ground = S.singleton (-1) -- Singleton for bricks supported only by the ground
go fall cur | z == 0 = M.insert i (cur { minZ=z+1 , maxZ=mz+1 , cubes=backCubes, supporting=ground }) fall -- Touched the ground
| null touching = go fall (cur { minZ=z+dz, maxZ=mz+dz, cubes=nextCubes }) -- Touched no bricks
| otherwise = M.insert i (cur { minZ=z+1 , maxZ=mz+1 , cubes=backCubes, supporting=touching }) fall
where i = M.size fall -- The index for the current brick
z = minZ cur -- The bottom z value of the brick
mz = maxZ cur -- The top z value of the brick
-- Speeding up the process by going more that one step at a time
nextZ = maximum $ 0 : [maxZ c | c <- M.elems fall, maxZ c < z]
dz = nextZ - z
-- Find the supporting bricks at that z step
touching = M.keysSet . M.filter (not . S.disjoint (cubes cur) . cubes) $ fall
-- Get the cubes for going up or continuing falling done
backCubes = S.map (third3 succ ) . cubes $ cur
nextCubes = S.map (third3 (+dz)) . cubes $ cur
Here is now my parseInput function:
parseInput :: String -> Input
parseInput = fallDown . sort . map read . lines
Part one is about finding which bricks can be safely removed. A brick can be removed if:
That is:
My claim is that, with the way I represent my data, it is easier to find bricks that are NOT safe to remove.
Then to find the ones that are safe to remove, we simply need to take the complement set of our bricks (and because we only care about sizes, we actually need to subtract the number of total bricks by the number of unremovable bricks)
To find unremovable bricks, I simply go through all of my bricks:
This, however, will also put the ground as an unremovable brick (which is, ultimately, not false, however it would make things more annoying to work with), therefore I remove it from my set.
getUnremovableBricks :: Input -> S.Set Int
getUnremovableBricks = S.delete (-1) . M.foldr go S.empty
where go cur acc | (S.size . supporting) cur == 1 = acc `S.union` supporting cur
| otherwise = acc
Now, I can simply compare sizes to get my answer:
partOne :: Input -> Output
partOne input = M.size input - (S.size . getUnremovableBricks) input
Now that we know which bricks we shouldn’t remove, as they would make other bricks fall, let’s remove them to make other bricks fall! :3c
First of all, I want a way to remove a brick from my bricks:
-- Remove a set of bricks from the bricks
removeBricks :: S.Set Int -> Input -> Input
removeBricks toRemove = M.map removeSupporting . (`M.withoutKeys` toRemove)
where removeSupporting c = c { supporting=S.difference (supporting c) toRemove }
Now that I can remove bricks from the tower, I need to know which bricks will now be falling.
A brick is falling if:
So, I simply need to check that:
-- Get the bricks that are no longer being supported
getFalling :: Input -> S.Set Int
getFalling = M.keysSet . M.filter (S.null . supporting)
Finally, I need to remove the falling bricks until no other brick is falling. I do that using a fix point algorithm:
This last part may sound counter-intuitive: the brick is not being removed, it is simply falling.
The reason it works is because we don’t actually need to care about where the brick is going to land. All we care about is that, for a short amount of time, this brick would not be supporting any other brick (which may cause them to fall as well)
Another way to think about it is by saying that removing a brick from the tower is actually marking it as “has fallen”.
-- Remove a brick and let the chain reaction happen
disintegrate :: Input -> Int -> Input
disintegrate bricks removed = until (null . getFalling) removeFalling startState -- Fix-point iteration
where toRemove = S.singleton removed
startState = removeBricks toRemove bricks
removeFalling = removeBricks =<< getFalling
And now, to solve part two:
partTwo :: Input -> Output
partTwo input = sum fallenCount
where toRemove = (S.toList . getUnremovableBricks) input
disintegrations = parMap rseq (disintegrate input) toRemove
totalBricks = M.size input
fallenCount = map (subtract 1 . (totalBricks -) . M.size) disintegrations