adventofcode/2020/11/main.py
2020-12-11 08:15:48 +01:00

89 lines
1.7 KiB
Python

import copy
from pprint import pprint
class PuzzleSolver:
def __init__(self, filename):
self.load(filename)
def load(self, filename):
with open(filename, "r", encoding="utf-8") as f:
self.data = [list(x.rstrip()) for x in f.readlines()]
self.width = len(self.data[0])
self.height = len(self.data)
def is_adjacent(self, x1, y1, x2, y2):
dy = abs(y2 - y1)
dx = abs(x2 - x1)
return dy <= 1 and dx <= 1
def occupied_adjacent(self, x, y):
n = 0
for j in range(y-1, y+2):
for i in range(x-1, x+2):
if (i, j) == (x, y):
continue
if not (0 <= i < self.width):
continue
if not (0 <= j < self.height):
continue
s = self.get(i, j)
if s == "#":
n += 1
return n
def get(self, x, y):
return self.data[y][x]
def mutate(self):
ret = copy.deepcopy(self.data)
for y in range(self.height):
for x in range(self.width):
s = self.get(x, y)
if s == ".":
continue
n = self.occupied_adjacent(x, y)
if s == "L":
if n == 0:
ret[y][x] = "#"
elif s == "#":
if n >= 4:
ret[y][x] = "L"
return ret
def print(self):
for y in range(self.height):
print("".join(self.data[y]))
print()
def solve(self):
stable = False
rounds = 0
self.print()
while not stable:
x = self.mutate()
stable = x == self.data
self.data = x
rounds += 1
self.print()
return rounds
def count_occupied(self):
return sum([x.count("#") for x in self.data])
ps = PuzzleSolver("test.txt")
rounds = ps.solve()
cs = PuzzleSolver("test-1.txt")
assert ps.data == cs.data
ps = PuzzleSolver("data.txt")
rounds = ps.solve()
n = ps.count_occupied()
print(f"There are {n} occupied seats.")