-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathaoc23_day10_1.py
executable file
·85 lines (64 loc) · 1.75 KB
/
aoc23_day10_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
80
81
82
83
84
85
#!/usr/bin/env python
def read_tubes():
res = []
while True:
try:
res.append(input().strip())
except EOFError:
break
return res
def find_start(tubes):
for i in range(len(tubes)):
for j in range(len(tubes[0])):
if tubes[i][j] == 'S':
return i, j
raise Exception('should find start')
NEXT_STEP = {
(1, 0, '|'): (1, 0),
(-1, 0, '|'): (-1, 0),
(0, 1, '-'): (0, 1),
(0, -1, '-'): (0, -1),
(0, -1, 'L'): (-1, 0),
(1, 0, 'L'): (0, 1),
(0, 1, 'J'): (-1, 0),
(1, 0, 'J'): (0, -1),
(0, 1, '7'): (1, 0),
(-1, 0, '7'): (0, -1),
(-1, 0, 'F'): (0, 1),
(0, -1, 'F'): (1, 0)
}
def get_next_step(last, c):
sr, sc = last
return NEXT_STEP.get((sr, sc, c))
def follow_track(step, current, tubes):
r, c = current
path = []
visited = set()
while True:
if r < 0 or r >= len(tubes) or c < 0 or c >= len(tubes[0]):
return False, []
if tubes[r][c] == '.':
return False, []
if tubes[r][c] == 'S':
return True, path
if (r, c) in visited:
return False, path
path.append((r, c))
visited.add((r, c))
step = get_next_step(step, tubes[r][c])
if step is None:
return False, []
r += step[0]
c += step[1]
def main():
tubes = read_tubes()
s_pos = find_start(tubes)
for step in [(1, 0), (-1,0), (0, 1), (0, -1)]:
ok, result = follow_track(step,
(s_pos[0] + step[0], s_pos[1] + step[1]),
tubes)
if ok:
print(f'{(len(result) + 1) // 2}')
break
if __name__ == '__main__':
main()