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.
163 lines
4.7 KiB
Elixir
163 lines
4.7 KiB
Elixir
1 month ago
|
defmodule Day9 do
|
||
|
def run do
|
||
|
memory =
|
||
|
File.read!("data.txt")
|
||
|
|> String.trim()
|
||
|
|> String.graphemes()
|
||
|
|> Enum.map(&String.to_integer(&1))
|
||
|
|> Enum.with_index()
|
||
|
|> Enum.reduce([], fn {num, idx}, acc ->
|
||
|
new_num = if rem(idx, 2) == 0, do: trunc(idx / 2), else: nil
|
||
|
Enum.concat(acc, List.duplicate(new_num, num))
|
||
|
end)
|
||
|
|
||
|
dbg(memory)
|
||
|
|
||
|
part1 = memory |> defrag1() |> checksum()
|
||
|
dbg(part1)
|
||
|
|
||
|
part2 = memory |> defrag2() |> checksum()
|
||
|
dbg(part2)
|
||
|
end
|
||
|
|
||
|
defp checksum(memory) do
|
||
|
Enum.with_index(memory)
|
||
|
|> Enum.reduce(0, fn {el, idx}, acc ->
|
||
|
if el, do: acc + idx * el, else: acc
|
||
|
end)
|
||
|
end
|
||
|
|
||
|
defp defrag1(memory) do
|
||
|
defrag1_step(%{memory: memory, free_index: 0, block_index: length(memory) - 1})
|
||
|
end
|
||
|
|
||
|
# 2 pointers, one tracking free space, one tracking blocks
|
||
|
# increase the free one until we find a spot, decrease the block one until we find a block
|
||
|
# if we have a free spot and a block, swap em
|
||
|
defp defrag1_step(acc) do
|
||
|
free = Enum.at(acc.memory, acc.free_index)
|
||
|
block = Enum.at(acc.memory, acc.block_index)
|
||
|
|
||
|
# if we've met in the middle, we're done recursing
|
||
|
if acc.free_index >= acc.block_index do
|
||
|
acc.memory
|
||
|
else
|
||
|
case {free, block} do
|
||
|
{_, nil} ->
|
||
|
%{acc | block_index: acc.block_index - 1}
|
||
|
|
||
|
{f, _} when f != nil ->
|
||
|
%{acc | free_index: acc.free_index + 1}
|
||
|
|
||
|
{nil, block} ->
|
||
|
memory =
|
||
|
List.replace_at(
|
||
|
List.replace_at(acc.memory, acc.free_index, block),
|
||
|
acc.block_index,
|
||
|
nil
|
||
|
)
|
||
|
|
||
|
%{
|
||
|
acc
|
||
|
| memory: memory,
|
||
|
block_index: acc.block_index - 1,
|
||
|
free_index: acc.free_index + 1
|
||
|
}
|
||
|
end
|
||
|
|> defrag1_step()
|
||
|
end
|
||
|
end
|
||
|
|
||
|
defp defrag2(memory) do
|
||
|
# break the memory up into chunks. Each time the char changes, new chunk
|
||
|
chunks =
|
||
|
Enum.chunk_while(
|
||
|
memory,
|
||
|
[],
|
||
|
fn el, acc ->
|
||
|
if length(acc) >= 1 and el != hd(acc) do
|
||
|
{:cont, Enum.reverse(acc), [el]}
|
||
|
else
|
||
|
{:cont, [el | acc]}
|
||
|
end
|
||
|
end,
|
||
|
fn
|
||
|
[] -> {:cont, []}
|
||
|
acc -> {:cont, Enum.reverse(acc), []}
|
||
|
end
|
||
|
)
|
||
|
|
||
|
# run it through the defragger and then flatten it
|
||
|
defrag2_step(%{chunks: chunks, index: length(chunks) - 1})
|
||
|
|> List.flatten()
|
||
|
end
|
||
|
|
||
|
defp defrag2_step(args) do
|
||
|
chunks = args.chunks
|
||
|
index = args.index
|
||
|
|
||
|
# if we are at 0, stop recursing
|
||
|
if index == 0 do
|
||
|
chunks
|
||
|
else
|
||
|
file = Enum.at(chunks, index)
|
||
|
|
||
|
# if this file is nils (empty space) skip it. recruse, picking next lowest chunk
|
||
|
if is_nil(Enum.at(file, 0)) do
|
||
|
defrag2_step(%{chunks: chunks, index: index - 1})
|
||
|
else
|
||
|
file_size = length(file)
|
||
|
|
||
|
# find a chunk of empty space big enough starting from the beginning
|
||
|
empty_chunk_index =
|
||
|
Enum.find_index(chunks, fn chunk ->
|
||
|
# if it's an empty space chunk and its big enough
|
||
|
is_nil(Enum.at(chunk, 0)) and length(chunk) >= file_size
|
||
|
end)
|
||
|
|
||
|
if empty_chunk_index && empty_chunk_index < index do
|
||
|
empty_chunk = Enum.at(chunks, empty_chunk_index)
|
||
|
new_free_space_size = length(empty_chunk) - file_size
|
||
|
|
||
|
# first replace the file with free space
|
||
|
temp1 =
|
||
|
List.replace_at(chunks, index, List.duplicate(nil, file_size))
|
||
|
|> List.replace_at(empty_chunk_index, file)
|
||
|
|
||
|
# if our file was smaller than the free space
|
||
|
if new_free_space_size > 0 do
|
||
|
# check the chunk _after_ the free space.
|
||
|
next_chunk = Enum.at(chunks, empty_chunk_index + 1)
|
||
|
|
||
|
new_list =
|
||
|
if is_nil(Enum.at(next_chunk, 0)) do
|
||
|
# if it is also free space, tack this on to it
|
||
|
List.replace_at(
|
||
|
temp1,
|
||
|
empty_chunk_index + 1,
|
||
|
Enum.concat(next_chunk, List.duplicate(nil, new_free_space_size))
|
||
|
)
|
||
|
else
|
||
|
# otherwise, create a new chunk after this one, filling up the space
|
||
|
List.insert_at(
|
||
|
temp1,
|
||
|
empty_chunk_index + 1,
|
||
|
List.duplicate(nil, new_free_space_size)
|
||
|
)
|
||
|
end
|
||
|
|
||
|
defrag2_step(%{chunks: new_list, index: index})
|
||
|
else
|
||
|
defrag2_step(%{chunks: temp1, index: index})
|
||
|
end
|
||
|
else
|
||
|
# no free space for this file. recurse, check the next chunk
|
||
|
defrag2_step(%{chunks: chunks, index: index - 1})
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
Day9.run()
|