Solutions and Write-Ups for my Advent Of Code adventures (mainly in Haskell)
That was the craziest part 2 so far Owo
Once again, a grid of char, once again I used a Data.Array.
I am not going to explain much more how I parsed it, it’s pretty similar to previous days, especially Day 14.
Note that I compiled the input inside a data structure:
data World = World { grid :: Array (Int, Int) Char, start :: (Int, Int), end :: (Int, Int) } deriving Show
parseInput :: String -> Input
parseInput input = World grid start end
where gridLines = lines input
(height, width) = (length gridLines, length $ head gridLines)
grid = listArray ((1, 1), (height, width)) $ filter (/= '\n') input
start = fst . head . filter ((== 'S') . snd) . assocs $ grid
end = fst . head . filter ((== 'E') . snd) . assocs $ grid
We need to find the length of the shortest path going from start to end.
Let’s use a shortest path algorithm from the Algorithm.Search library.
I have decided to use dijkstra, it is well suited here because we can model our input as a weighted directed graph with positive weights.
In order to use that function we need to define a few things:
I defined a state as a position and a direction:
data State = State { position :: (Int, Int), direction :: (Int, Int) } deriving (Show, Eq, Ord)
The neighbours of a state are:
The tile has to be a non-wall tile inside the grid.
applyTuple :: (a -> b -> c) -> (a, a) -> (b, b) -> (c, c)
applyTuple f (a, b) (c, d) = (f a c, f b d)
getRotations :: (Int, Int) -> [(Int, Int)]
getRotations (di, dj) = [(di, dj), (dj, di), (-dj, -di)]
getNextStates :: Array (Int, Int) Char -> State -> [State]
getNextStates grid (State pos dir) = nexts
where rotations = getRotations dir
steps = [State (applyTuple (+) pos d) d | d <- rotations]
nexts = filter ((/= '#') . (grid !) . position) steps
Now that I can get the neighbour of a tile, I need to define the cost function:
Assuming that s1 and s2 are neighbours
getCost :: State -> State -> Int
getCost s1 s2 | direction s1 /= direction s2 = 1001
| otherwise = 1
Now part 1 is just calling the dijkstra function with our model:
partOne :: Input -> Output
partOne (World arr start end) = fst . fromJust $ dijkstra getNexts getCost isDone (State start (0, 1))
where getNexts = getNextStates arr
isDone = (== end) . position
The path that the dijkstra function found is not the only shortest path.
Let’s find all the tiles that are part of a shortest path.
Omg.
Alright so, the easy way is to modify the dijkstra function to find all the shortest paths, as written on the Wikipedia page
However, I used a library, and I do not want to code Dijkstra in Haskell.
So what I did is the following:
dfs x mem dist | position x == end = Map.insert x (True, dist) mem
| dist >= bestDist = mem
| otherwise = mem''
where (_, bestDist) = Map.findWithDefault (False, target) x mem
neighbours = getNextStates arr x
costs = map (getCost x) neighbours
tryVisit (next, cost) mem = dfs next mem (dist + cost)
isNeighbourOnPath cost (onPath, neighDist) = onPath && neighDist == dist + cost
mem' = foldr tryVisit mem (zip neighbours costs)
mem'' | or $ zipWith isNeighbourOnPath costs [Map.findWithDefault (False, target) n mem' | n <- neighbours] = Map.insert x (True, dist) mem'
| otherwise = Map.insert x (False, dist) mem'
Once we have that, we can filter the map to only get the states that are part of the path.
Then, we get the coordinates of these states (and we only keep unique ones), before counting how many tiles there are:
partTwo :: Input -> Output
partTwo input@(World arr start end) = length . nub . map position . Map.keys $ statesOnPath
where target = partOne input
states = dfs (State start (0, 1)) Map.empty 0
statesOnPath = Map.filter fst states
dfs x mem dist | position x == end = Map.insert x (True, dist) mem
| dist >= bestDist = mem
| otherwise = mem''
where (_, bestDist) = Map.findWithDefault (False, target) x mem
neighbours = getNextStates arr x
costs = map (getCost x) neighbours
tryVisit (next, cost) mem = dfs next mem (dist + cost)
isNeighbourOnPath cost (onPath, neighDist) = onPath && neighDist == dist + cost
mem' = foldr tryVisit mem (zip neighbours costs)
mem'' | or $ zipWith isNeighbourOnPath costs [Map.findWithDefault (False, target) n mem' | n <- neighbours] = Map.insert x (True, dist) mem'
| otherwise = Map.insert x (False, dist) mem'
I have a feeling that I just recoded some path-finding algorithm, but to be honest I have no idea what I did here, if it even has a name.
In fact, this can solve both part one and two.
It seems to be a path-finding algorithm that works well when given a good higher-bound for distances.
One higher bound that can be easily found is 1001 * taxicab start end:
partBonus :: Input -> (Output, Output)
partBonus input@(World arr start end) = (distOfEnd, tilesOnPath)
where distance (i, j) (k, l) = abs (i - k) + abs (j - l)
maxDist = 1001 * distance start end
states = dfs (State start (0, 1)) Map.empty 0
statesOnPath = Map.filter fst states
distOfEnd = snd . Map.findMin . Map.map snd $ Map.filterWithKey (\k _ -> position k == end) states
tilesOnPath = length . nub . map position . Map.keys $ statesOnPath
dfs x mem dist | position x == end = Map.insert x (True, dist) mem
| dist >= bestDist = mem
| otherwise = mem''
where (_, bestDist) = Map.findWithDefault (False, maxDist) x mem
neighbours = getNextStates arr x
costs = map (getCost x) neighbours
tryVisit (next, cost) mem = dfs next mem (dist + cost)
isNeighbourOnPath cost (onPath, neighDist) = onPath && neighDist == dist + cost
mem' = foldr tryVisit mem (zip neighbours costs)
mem'' | or $ zipWith isNeighbourOnPath costs [Map.findWithDefault (False, maxDist) n mem' | n <- neighbours] = Map.insert x (True, dist) mem'
| otherwise = Map.insert x (False, dist) mem'
Comparisons:
➜ Day_16 git:(main) ✗ time ./Day_16 one two input
134588
631
./Day_16 one two input 1.26s user 0.05s system 97% cpu 1.337 total
➜ Day_16 git:(main) ✗ time ./Day_16 bonus input
(134588,734)
./Day_16 bonus input 4.74s user 0.07s system 98% cpu 4.869 total
I also have a feeling that it is technicaly not right as a path-finding algorithm:
One way to fix that would be to swap the position x == end
guard with the dist >= bestDist
guard, and to initialise mem with all possible states for end at (True, maxDist)
But even that won’t fix everything. I know that if the higher bound is not exactly the shortest path length some input won’t work.
I will probably analyze my algorithm further in the upcoming months, because I feel like it’s quite fascinating and I want to understand it better and improve it. It was made in a few dozen minutes, therefore it is quite unpolished and unoptimised, and only works on intuition rather than concrete proof, but I feel like there is something interesting in it.
If anyone know what the hell I did, don’t hesitate to reach me on bluesky.