-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path19.go
118 lines (110 loc) · 2.96 KB
/
19.go
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
111
112
113
114
115
116
117
118
package main
import (
"bufio"
"fmt"
"os"
"sort"
)
func main() {
blueprints := []blueprint{}
scanner := bufio.NewScanner(os.Stdin)
for scanner.Scan() {
var b blueprint
fmt.Sscanf(scanner.Text(),
"Blueprint %d: Each ore robot costs %d ore. Each clay robot costs %d ore. Each obsidian robot costs %d ore and %d clay. Each geode robot costs %d ore and %d obsidian.",
&b.n, &b.ore, &b.clay, &b.obsidian.ore, &b.obsidian.clay, &b.geode.ore, &b.geode.obsidian)
blueprints = append(blueprints, b)
}
sum := 0
for _, bp := range blueprints {
sum += bp.n * maxgeodes(bp, 24, 2000000)
}
prod := 1
for _, bp := range blueprints[:3] {
prod *= maxgeodes(bp, 32, 2000000)
}
fmt.Println(sum, prod)
}
type blueprint struct {
n int
ore, clay int
obsidian struct{ ore, clay int }
geode struct{ ore, obsidian int }
}
type items struct{ ore, clay, obsidian, geode int }
func (i items) add(ore, clay, obsidian, geode int) items {
return items{i.ore + ore, i.clay + clay, i.obsidian + obsidian, i.geode + geode}
}
type state struct {
minute int
values items
robots items
}
func (s state) val() int {
return s.robots.ore + s.robots.clay*10 + s.robots.obsidian*100 + s.robots.geode*1000
}
func maxgeodes(bp blueprint, minutes, maxlen int) int {
max := 0
maxmin := 0
processed := make(map[state]bool)
queue := []state{{0, items{0, 0, 0, 0}, items{1, 0, 0, 0}}}
for len(queue) > 0 {
if queue[0].minute > maxmin {
maxmin = queue[0].minute
processed = map[state]bool{}
if len(queue) > maxlen {
sort.Slice(queue, func(i, j int) bool {
return queue[i].val() > queue[j].val()
})
queue = queue[0:maxlen]
continue
}
}
s := queue[0]
queue = queue[1:]
if s.minute == minutes {
if s.values.geode > max {
max = s.values.geode
}
continue
}
if processed[s] {
continue
}
processed[s] = true
if s.values.ore >= bp.geode.ore && s.values.obsidian >= bp.geode.obsidian {
queue = append(queue, state{
s.minute + 1,
s.values.add(s.robots.ore-bp.geode.ore, s.robots.clay, s.robots.obsidian-bp.geode.obsidian, s.robots.geode),
s.robots.add(0, 0, 0, 1),
})
}
if s.values.ore >= bp.obsidian.ore && s.values.clay >= bp.obsidian.clay {
queue = append(queue, state{
s.minute + 1,
s.values.add(s.robots.ore-bp.obsidian.ore, s.robots.clay-bp.obsidian.clay, s.robots.obsidian, s.robots.geode),
s.robots.add(0, 0, 1, 0),
})
}
if s.values.ore >= bp.clay {
queue = append(queue, state{
s.minute + 1,
s.values.add(s.robots.ore-bp.clay, s.robots.clay, s.robots.obsidian, s.robots.geode),
s.robots.add(0, 1, 0, 0),
})
}
if s.values.ore >= bp.ore {
queue = append(queue, state{
s.minute + 1,
s.values.add(s.robots.ore-bp.ore, s.robots.clay, s.robots.obsidian, s.robots.geode),
s.robots.add(1, 0, 0, 0),
})
}
queue = append(queue, state{
s.minute + 1,
s.values.add(s.robots.ore, s.robots.clay, s.robots.obsidian, s.robots.geode),
s.robots,
})
}
return max
}