-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain_day_12_1.py
79 lines (64 loc) · 2.14 KB
/
main_day_12_1.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
import sys
import os
from collections import deque
try:
from utils import measure_time
except ImportError:
project_root = os.path.abspath(
os.path.join(os.path.dirname(__file__), ".."))
sys.path.insert(0, project_root)
from utils import measure_time
CURRENT_DIR = os.path.dirname(__file__)
INPUT_FILE = os.path.join(CURRENT_DIR, "input.txt")
def calculate_area(region: set[tuple[int, int]]) -> int:
return len(region)
def calculate_perimeter(region: set[tuple[int, int]]) -> int:
total = 0
for r, c in region:
num_neighbors = len([
1 for dr, dc in ((-1, 0), (1, 0), (0, -1), (0, 1))
if (r + dr, c + dc) in region
])
total += 4 - num_neighbors
return total
def parse_regions(grid: list[str]) -> list[set[tuple[int, int]]]:
num_rows = len(grid)
num_cols = len(grid[0])
regions = []
seen = set()
for r in range(num_rows):
for c in range(num_cols):
if (r, c) in seen:
continue
region = set()
queue = deque([(r, c)])
while queue:
rr, cc = queue.popleft()
region.add((rr, cc))
for dr, dc in ((-1, 0), (1, 0), (0, -1), (0, 1)):
nr, nc = rr + dr, cc + dc
if (
(nr, nc) not in seen and
0 <= nr < num_rows and
0 <= nc < num_cols and
grid[nr][nc] == grid[rr][cc]
):
queue.append((nr, nc))
seen.add((nr, nc))
regions.append(region)
return regions
@measure_time
def calculate_total_fencing_cost(input_file: str):
with open(input_file, "r") as f:
grid = list(map(str.strip, f.readlines()))
regions = parse_regions(grid)
return sum(calculate_area(r) * calculate_perimeter(r) for r in regions)
if __name__ == "__main__":
try:
result = calculate_total_fencing_cost(INPUT_FILE)
print(result)
except Exception as e:
print(f"An error occurred: {e}")
sys.exit(1)
else:
sys.exit(0)