1
1
advent-of-code/2023/day3.py

54 lines
1.5 KiB
Python
Raw Normal View History

2023-12-03 08:09:25 +00:00
import string
import sys
from collections import defaultdict
2023-12-04 18:30:44 +00:00
NOT_A_SYMBOL = "." + string.digits
lines = sys.stdin.read().splitlines()
2023-12-03 08:09:25 +00:00
values: list[int] = []
gears: dict[tuple[int, int], list[int]] = defaultdict(list)
for i, line in enumerate(lines):
j = 0
while j < len(line):
# skip everything until a digit is found (start of a number)
if line[j] not in string.digits:
j += 1
continue
# extract the range of the number and its value
k = j + 1
while k < len(line) and line[k] in string.digits:
k += 1
value = int(line[j:k])
# lookup around the number if there is a symbol - we go through the number
# itself but that should not matter since it only contains digits
found = False
for i2 in range(max(0, i - 1), min(i + 1, len(lines) - 1) + 1):
for j2 in range(max(0, j - 1), min(k, len(line) - 1) + 1):
assert i2 >= 0 and i2 < len(lines)
assert j2 >= 0 and j2 < len(line)
2023-12-04 18:30:44 +00:00
if lines[i2][j2] not in NOT_A_SYMBOL:
2023-12-03 08:09:25 +00:00
found = True
if lines[i2][j2] == "*":
gears[i2, j2].append(value)
if found:
values.append(value)
# continue starting from the end of the number
j = k
2023-12-04 18:30:44 +00:00
# part 1
2023-12-03 08:09:25 +00:00
answer_1 = sum(values)
print(f"answer 1 is {answer_1}")
2023-12-04 18:30:44 +00:00
# part 2
answer_2 = sum(v1 * v2 for v1, v2 in filter(lambda vs: len(vs) == 2, gears.values()))
2023-12-03 08:09:25 +00:00
print(f"answer 2 is {answer_2}")