149 lines
4.1 KiB
Python
149 lines
4.1 KiB
Python
from lib import *
|
|
|
|
EXAMPLE = """px{a<2006:qkq,m>2090:A,rfg}
|
|
pv{a>1716:R,A}
|
|
lnx{m>1548:A,A}
|
|
rfg{s<537:gd,x>2440:R,A}
|
|
qs{s>3448:A,lnx}
|
|
qkq{x<1416:A,crn}
|
|
crn{x>2662:A,R}
|
|
in{s<1351:px,qqz}
|
|
qqz{s>2770:qs,m<1801:hdj,R}
|
|
gd{a>3333:R,R}
|
|
hdj{m>838:A,pv}
|
|
|
|
{x=787,m=2655,a=1222,s=2876}
|
|
{x=1679,m=44,a=2067,s=496}
|
|
{x=2036,m=264,a=79,s=2244}
|
|
{x=2461,m=1339,a=466,s=291}
|
|
{x=2127,m=1623,a=2188,s=1013}
|
|
"""
|
|
|
|
def solve(i: Input, second=False):
|
|
res = 0
|
|
ps = i.paras()
|
|
wfs, parts = ps
|
|
wfs = wfs.splitlines()
|
|
parts = parts.splitlines()
|
|
|
|
workflows = {}
|
|
for w in wfs:
|
|
name, ins = w.split("{")
|
|
ins = ins.replace("}", "")
|
|
conds = ins.split(",")
|
|
wf = []
|
|
for cond in conds[:-1]:
|
|
cmp, n = cond.split(":")
|
|
if '<' in cmp:
|
|
a, b = cmp.split('<')
|
|
wf.append((a, 'smaller', int(b), n))
|
|
elif '>' in cmp:
|
|
a, b = cmp.split('>')
|
|
wf.append((a, 'greater', int(b), n))
|
|
else:
|
|
raise Exception()
|
|
wf.append(conds[-1:])
|
|
workflows[name] = wf
|
|
|
|
def parse_part(part):
|
|
d = {}
|
|
p = part.replace("{", "").replace("}", "")
|
|
for pair in p.split(","):
|
|
a, b = pair.split("=")
|
|
d[a] = int(b)
|
|
return d
|
|
|
|
parts = list(map(parse_part, parts))
|
|
if not second:
|
|
for p in parts:
|
|
current = 'in'
|
|
while current not in ['A', 'R']:
|
|
for inst in workflows[current]:
|
|
if len(inst) == 4:
|
|
letter = inst[0]
|
|
value = inst[2]
|
|
next = inst[3]
|
|
if inst[1] == 'smaller':
|
|
if p[letter] < value:
|
|
current = next
|
|
break
|
|
elif inst[1] == 'greater':
|
|
if p[letter] > value:
|
|
current = next
|
|
break
|
|
else:
|
|
raise Exception()
|
|
elif len(inst) == 1:
|
|
current = inst[0]
|
|
else:
|
|
raise Exception()
|
|
if current == 'A':
|
|
r = sum(p.values())
|
|
res += r
|
|
return res
|
|
|
|
ranges = [{c: (1, 4000) for c in 'xmas'}]
|
|
ranges[0]['cur'] = 'in'
|
|
ranges[0]['idx'] = 0
|
|
accepted = []
|
|
while ranges:
|
|
r = ranges.pop()
|
|
cur, idx = r['cur'], r['idx']
|
|
|
|
if cur == 'A':
|
|
accepted.append(r)
|
|
continue
|
|
elif cur == 'R':
|
|
continue
|
|
|
|
inst = workflows[cur][idx]
|
|
if len(inst) == 4:
|
|
letter = inst[0]
|
|
value = inst[2]
|
|
nxt = inst[3]
|
|
ro = r[letter]
|
|
r1, r2 = dict(r), dict(r)
|
|
if inst[1] == 'smaller':
|
|
r1[letter] = (ro[0], value - 1)
|
|
r1['idx'] = 0
|
|
r1['cur'] = nxt
|
|
r2[letter] = (value, ro[1])
|
|
r2['idx'] += 1
|
|
elif inst[1] == 'greater':
|
|
r1[letter] = (ro[0], value)
|
|
r1['idx'] += 1
|
|
r2[letter] = (value + 1, ro[1])
|
|
r2['idx'] = 0
|
|
r2['cur'] = nxt
|
|
if r1[letter][1] >= r1[letter][0]:
|
|
ranges.append(r1)
|
|
if r2[letter][1] >= r2[letter][0]:
|
|
ranges.append(r2)
|
|
elif len(inst) == 1:
|
|
r['cur'] = inst[0]
|
|
r['idx'] = 0
|
|
ranges.append(r)
|
|
res = 0
|
|
for a in accepted:
|
|
r = 1
|
|
for c in 'xmas':
|
|
l, u = a[c]
|
|
r *= (u + 1 - l)
|
|
res += r
|
|
return res
|
|
|
|
def main():
|
|
DAY_INPUT = "i19.txt"
|
|
|
|
print("Example 1:", solve(Input(EXAMPLE)))
|
|
print("Solution 1:", solve(Input(DAY_INPUT)))
|
|
# 25:00
|
|
|
|
print("Example 2:", solve(Input(EXAMPLE), True))
|
|
assert solve(Input(EXAMPLE), True) == 167409079868000
|
|
print("Solution 2:", solve(Input(DAY_INPUT), True))
|
|
# 120:00
|
|
|
|
if __name__ == "__main__":
|
|
main()
|