2024-12-11 19:26:56 +00:00
|
|
|
package two
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
const blinks = 75
|
|
|
|
const magic = 2024
|
|
|
|
|
|
|
|
func Solve(buf []byte) (int, error) {
|
|
|
|
r := bytes.NewReader(buf)
|
|
|
|
numbers, err := parseLines(r)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
old := make(map[int]int)
|
|
|
|
for _, n := range numbers {
|
|
|
|
old[n]++
|
|
|
|
}
|
|
|
|
log.Debug(old)
|
|
|
|
|
|
|
|
memo := make(map[int]result)
|
|
|
|
for range blinks {
|
|
|
|
next := make(map[int]int)
|
|
|
|
|
|
|
|
for stone, count := range old {
|
|
|
|
var r result
|
2024-12-11 19:40:45 +00:00
|
|
|
r, ok := memo[stone]
|
2024-12-11 19:26:56 +00:00
|
|
|
if !ok {
|
|
|
|
r = applyRules(stone)
|
|
|
|
memo[stone] = r
|
|
|
|
}
|
|
|
|
|
|
|
|
switch r.kind {
|
|
|
|
case flip, multiply:
|
2024-12-11 19:40:45 +00:00
|
|
|
next[r.right] += count
|
2024-12-11 19:26:56 +00:00
|
|
|
case split:
|
2024-12-11 19:40:45 +00:00
|
|
|
next[r.left] += count
|
|
|
|
next[r.right] += count
|
2024-12-11 19:26:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
old = next
|
|
|
|
log.Debug(old)
|
|
|
|
}
|
|
|
|
|
|
|
|
var totalCount int
|
|
|
|
for _, count := range old {
|
|
|
|
totalCount += count
|
|
|
|
}
|
|
|
|
|
|
|
|
return totalCount, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyRules(num int) result {
|
|
|
|
switch num := num; {
|
|
|
|
case num == 0:
|
|
|
|
return result{flip, 0, 1}
|
|
|
|
case lenItoa(num)%2 == 0:
|
|
|
|
left, right := splitNumber(num)
|
|
|
|
return result{split, left, right}
|
|
|
|
default:
|
|
|
|
return result{multiply, 0, num * magic}
|
|
|
|
}
|
|
|
|
}
|