-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpart2.py
110 lines (85 loc) Β· 3.03 KB
/
part2.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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
from enum import Enum
from math import gcd
ModuleType = Enum("ModuleType", ["FlipFlop", "Conjunction", "Broadcaster"])
class Module:
def __init__(self, type, sends_to):
self.type = type
self.sends_to = sends_to
class FlipFlop(Module):
def __init__(self, sends_to):
super().__init__(ModuleType.FlipFlop, sends_to)
self.state = 0
class Conjunction(Module):
def __init__(self, sends_to):
super().__init__(ModuleType.Conjunction, sends_to)
self.inputs = {}
class Broadcaster(Module):
def __init__(self, sends_to):
super().__init__(ModuleType.Broadcaster, sends_to)
modules = {}
for line in open("input.txt"):
[id, sends_to_str] = line.strip().split(" -> ")
sends_to = sends_to_str.split(", ")
name = id[1:]
module = None
if id[0] == "%": module = FlipFlop(sends_to)
elif id[0] == "&": module = Conjunction(sends_to)
elif id == "broadcaster":
module = Broadcaster(sends_to)
name = id
modules[name] = module
def conjunction_find_senders(modules, conjunction_name):
for name, module in modules.items():
if name != conjunction_name and conjunction_name in module.sends_to:
modules[conjunction_name].inputs[name] = 0
for name, module in modules.items():
if module.type == ModuleType.Conjunction:
conjunction_find_senders(modules, name)
def broadcast(modules, n, map, count):
queue = [("button", "broadcaster", 0)]
while queue != []:
(sender, reciever, recieved) = queue.pop(0)
if recieved == 1 and sender in map:
if len(map[sender]) == 1:
count += 1
map[sender].append(n)
if count == len(map):
return (True, count)
to_send = None
if modules[reciever].type == ModuleType.Broadcaster:
to_send = recieved
elif modules[reciever].type == ModuleType.FlipFlop:
if recieved == 0:
modules[reciever].state = 1 - modules[reciever].state
to_send = modules[reciever].state
elif modules[reciever].type == ModuleType.Conjunction:
modules[reciever].inputs[sender] = recieved
to_send = 0 if list(modules[reciever].inputs.values()).count(0) == 0 else 1
if to_send == None:
continue
for to in modules[reciever].sends_to:
if to in modules:
queue.append((reciever, to, to_send))
elif to == "rx":
rx = to_send
return (False, count)
map = {}
count = 0
for _, module in modules.items():
if "rx" in module.sends_to and module.type == ModuleType.Conjunction:
for name in module.inputs:
map[name] = []
def lcm(ns):
result = 1
for n in ns:
result *= n // gcd(result, n)
return result
presses = None
n = 0
while True:
(result, count) = broadcast(modules, n + 1, map, count)
if result:
presses = lcm(list(v[1] - v[0] for name, v in map.items()))
break
n += 1
print(presses)