Compare commits
No commits in common. "f92e492f9be17c811ccdd7799e6f3a74245da432" and "e348cea55a68d0ab730e09bc377f53541be18e47" have entirely different histories.
f92e492f9b
...
e348cea55a
|
@ -1,62 +0,0 @@
|
||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
import sys
|
|
||||||
|
|
||||||
|
|
||||||
def solve(input: str) -> int:
|
|
||||||
def parse(input: str) -> tuple[list[int], list[int]]:
|
|
||||||
files: list[int] = []
|
|
||||||
free: list[int] = []
|
|
||||||
|
|
||||||
for i, c in enumerate(input):
|
|
||||||
if c == "\n":
|
|
||||||
continue
|
|
||||||
(files if i % 2 == 0 else free).append(int(c))
|
|
||||||
|
|
||||||
# Make `to_disk` slightly simpler
|
|
||||||
if len(files) > len(free):
|
|
||||||
free.append(0)
|
|
||||||
|
|
||||||
return files, free
|
|
||||||
|
|
||||||
def to_disk(files: list[int], free: list[int]) -> list[int | None]:
|
|
||||||
assert len(files) == len(free) # Sanity check
|
|
||||||
|
|
||||||
disk: list[int | None] = []
|
|
||||||
|
|
||||||
for i in range(len(files)):
|
|
||||||
disk.extend([i for _ in range(files[i])])
|
|
||||||
disk.extend([None for _ in range(free[i])])
|
|
||||||
|
|
||||||
return disk
|
|
||||||
|
|
||||||
def compact(disk: list[int | None]) -> list[int]:
|
|
||||||
last = len(disk) - 1
|
|
||||||
for i in range(len(disk)):
|
|
||||||
# Are we finished compacting?
|
|
||||||
if i >= last:
|
|
||||||
break
|
|
||||||
# Is this a free space?
|
|
||||||
if disk[i] is not None:
|
|
||||||
continue
|
|
||||||
assert disk[last] is not None # Sanity check
|
|
||||||
disk[i], disk[last] = disk[last], disk[i]
|
|
||||||
# Find next block to compact
|
|
||||||
while disk[last] is None:
|
|
||||||
last -= 1
|
|
||||||
# At this point, `None` are at the end
|
|
||||||
# Naive list comprehension to appease the type checker
|
|
||||||
return [n for n in disk if n is not None]
|
|
||||||
|
|
||||||
files, free = parse(input)
|
|
||||||
disk = to_disk(files, free)
|
|
||||||
return sum(i * n for i, n in enumerate(compact(disk)))
|
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
|
||||||
input = sys.stdin.read()
|
|
||||||
print(solve(input))
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
File diff suppressed because one or more lines are too long
|
@ -1,93 +0,0 @@
|
||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
import sys
|
|
||||||
from typing import NamedTuple
|
|
||||||
|
|
||||||
|
|
||||||
class FilePosition(NamedTuple):
|
|
||||||
pos: int
|
|
||||||
length: int
|
|
||||||
|
|
||||||
|
|
||||||
class ParsedDisk(NamedTuple):
|
|
||||||
files: list[FilePosition]
|
|
||||||
holes: list[FilePosition]
|
|
||||||
|
|
||||||
|
|
||||||
def solve(input: str) -> int:
|
|
||||||
def parse(input: str) -> tuple[list[int], list[int]]:
|
|
||||||
files: list[int] = []
|
|
||||||
free: list[int] = []
|
|
||||||
|
|
||||||
for i, c in enumerate(input):
|
|
||||||
if c == "\n":
|
|
||||||
continue
|
|
||||||
(files if i % 2 == 0 else free).append(int(c))
|
|
||||||
|
|
||||||
# Make `to_disk` slightly simpler
|
|
||||||
if len(files) > len(free):
|
|
||||||
free.append(0)
|
|
||||||
|
|
||||||
return files, free
|
|
||||||
|
|
||||||
def to_disk(files: list[int], free: list[int]) -> ParsedDisk:
|
|
||||||
assert len(files) == len(free) # Sanity check
|
|
||||||
|
|
||||||
pos = 0
|
|
||||||
disk_files: list[FilePosition] = []
|
|
||||||
holes: list[FilePosition] = []
|
|
||||||
|
|
||||||
for i in range(len(files)):
|
|
||||||
disk_files.append(FilePosition(pos, files[i]))
|
|
||||||
pos += files[i]
|
|
||||||
holes.append(FilePosition(pos, free[i]))
|
|
||||||
pos += free[i]
|
|
||||||
|
|
||||||
return ParsedDisk(disk_files, holes)
|
|
||||||
|
|
||||||
def compact(disk: ParsedDisk) -> list[int | None]:
|
|
||||||
def move_files(disk: ParsedDisk) -> dict[int, FilePosition]:
|
|
||||||
new_files: dict[int, FilePosition] = {}
|
|
||||||
for i, (pos, length) in reversed(list(enumerate(disk.files))):
|
|
||||||
for h in range(len(disk.holes)):
|
|
||||||
# Hole must be big enough to fit the file
|
|
||||||
if length > disk.holes[h].length:
|
|
||||||
continue
|
|
||||||
# Hole must be to the left of the file
|
|
||||||
if pos < disk.holes[h].pos:
|
|
||||||
break
|
|
||||||
# We found a hole, move the file into it
|
|
||||||
pos = disk.holes[h].pos
|
|
||||||
disk.holes[h] = FilePosition(
|
|
||||||
length=disk.holes[h].length - length,
|
|
||||||
pos=disk.holes[h].pos + length,
|
|
||||||
)
|
|
||||||
break
|
|
||||||
new_files[i] = FilePosition(pos, length)
|
|
||||||
return new_files
|
|
||||||
|
|
||||||
def to_disk(files: dict[int, FilePosition]) -> list[int | None]:
|
|
||||||
disk: list[int | None] = [None] * max(
|
|
||||||
(f.pos + f.length) for f in files.values()
|
|
||||||
)
|
|
||||||
for i, f in files.items():
|
|
||||||
for j in range(f.length):
|
|
||||||
assert disk[f.pos + j] is None # Sanity check
|
|
||||||
disk[f.pos + j] = i
|
|
||||||
return disk
|
|
||||||
|
|
||||||
new_files = move_files(disk)
|
|
||||||
return to_disk(new_files)
|
|
||||||
|
|
||||||
files, free = parse(input)
|
|
||||||
disk = to_disk(files, free)
|
|
||||||
return sum(i * n for i, n in enumerate(compact(disk)) if n is not None)
|
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
|
||||||
input = sys.stdin.read()
|
|
||||||
print(solve(input))
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
main()
|
|
File diff suppressed because one or more lines are too long
Loading…
Reference in a new issue