You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
62 lines
1.6 KiB
Elixir
62 lines
1.6 KiB
Elixir
defmodule Day10 do
|
|
def run do
|
|
map =
|
|
File.read!("data.txt")
|
|
|> String.trim()
|
|
|> String.split("\n")
|
|
|> Enum.map(fn el -> String.graphemes(el) |> Enum.map(&String.to_integer/1) end)
|
|
|
|
height = length(map)
|
|
width = length(hd(map))
|
|
|
|
trailheads =
|
|
Enum.concat(map)
|
|
|> Enum.with_index()
|
|
|> Enum.filter(fn {el, _} -> el == 0 end)
|
|
|> Enum.map(fn {_, idx} -> {div(idx, height), rem(idx, width)} end)
|
|
|
|
paths1 =
|
|
Enum.reduce(trailheads, 0, fn th, acc ->
|
|
(walk(th, map) |> List.flatten() |> Enum.uniq() |> length) + acc
|
|
end)
|
|
|
|
dbg(paths1)
|
|
|
|
# same as part one, just don't uniq() them
|
|
paths2 =
|
|
Enum.reduce(trailheads, 0, fn th, acc ->
|
|
(walk(th, map) |> List.flatten() |> length) + acc
|
|
end)
|
|
|
|
dbg(paths2)
|
|
end
|
|
|
|
defp walk(coord, map) do
|
|
value = element_at(coord, map)
|
|
|
|
if value == 9 do
|
|
coord
|
|
else
|
|
# find the coords of all neighbors
|
|
neighbors(coord)
|
|
# filter by whether the element is 1 greater
|
|
|> Enum.filter(fn c -> element_at(c, map) == value + 1 end)
|
|
# add this path to the list. it'll be empty and then flattened away if it doesn't reach the top
|
|
|> Enum.reduce([], fn el, acc -> [walk(el, map) | acc] end)
|
|
end
|
|
end
|
|
|
|
defp neighbors({x, y}) do
|
|
[{-1, 0}, {0, 1}, {1, 0}, {0, -1}]
|
|
|> Enum.map(fn {dx, dy} -> {x + dx, y + dy} end)
|
|
|> Enum.reject(fn {x2, y2} -> x2 < 0 or y2 < 0 end)
|
|
end
|
|
|
|
defp element_at({x, y}, map) do
|
|
row = Enum.at(map, x)
|
|
if is_nil(row), do: nil, else: Enum.at(row, y)
|
|
end
|
|
end
|
|
|
|
Day10.run()
|