aboutsummaryrefslogtreecommitdiff
path: root/2023/17/solve.py
blob: 24e44715837f1aaa7b6406f3d8b2b710f8b8ad4d (plain) (blame)
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
from queue import PriorityQueue

L = {
    (y, x): int(c)
    for y, l in enumerate(open("input").readlines())
    for x, c in enumerate(l.strip())
}


W = max(x for (_, x) in L.keys())
H = max(y for (y, _) in L.keys())


def oob(y, x, d, c):
    return x > W or x < 0 or y > H or y < 0


def neighbours(y, x, d, c):
    ns = []
    if d != 2:
        ns.append((y - 1, x, 0, c + 1 if d == 0 else 0))
    if d != 3:
        ns.append((y, x + 1, 1, c + 1 if d == 1 else 0))
    if d != 0:
        ns.append((y + 1, x, 2, c + 1 if d == 2 else 0))
    if d != 1:
        ns.append((y, x - 1, 3, c + 1 if d == 3 else 0))
    return ns


def part1():
    q = PriorityQueue()
    q.put((0, (0, 0, 1, 0)))
    v = set()

    while q:
        p, c = q.get()

        if c[:2] == (H, W):
            break

        if c in v:
            continue
        v.add(c)

        for n in neighbours(*c):
            if not oob(*n) and c[3] < 3:
                q.put((p + L[n[:2]], n))

    print(p)


def part2():
    q = PriorityQueue()
    q.put((0, (0, 0, 1, 0)))
    v = set()

    while q:
        p, c = q.get()

        if c[:2] == (H, W) and c[3] >= 3:
            print(p)
            break

        if c in v:
            continue
        v.add(c)

        for n in neighbours(*c):
            if n[2] != c[2] and c[3] < 3:
                continue
            if not oob(*n) and n[3] < 10:
                q.put((p + L[n[:2]], n))


part1()
part2()