319 lines
8.2 KiB
Swift
319 lines
8.2 KiB
Swift
|
import Foundation
|
||
|
|
||
|
@main
|
||
|
struct Day18: Puzzle {
|
||
|
func run() {
|
||
|
let pairs = Scanner(string: input).all()
|
||
|
let first = pairs[0]
|
||
|
let result = pairs.dropFirst().reduce(first, +)
|
||
|
print(result.toString())
|
||
|
print("Part 1:", result.magnitude())
|
||
|
|
||
|
var largest = 0
|
||
|
for a in pairs {
|
||
|
for b in pairs where a !== b {
|
||
|
largest = max(largest, (a + b).magnitude())
|
||
|
largest = max(largest, (b + a).magnitude())
|
||
|
}
|
||
|
}
|
||
|
print("Part 2:", largest)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
extension Scanner {
|
||
|
func pair() -> Pair? {
|
||
|
guard scanString("[") != nil else { return nil }
|
||
|
guard let leftPart = part() else { return nil }
|
||
|
guard scanString(",") != nil else { return nil }
|
||
|
guard let rightPart = part() else { return nil }
|
||
|
guard scanString("]") != nil else { return nil }
|
||
|
return Pair(leftPart, rightPart)
|
||
|
}
|
||
|
|
||
|
func part() -> Number? {
|
||
|
if let digits = scanInt() {
|
||
|
return Regular(value: digits)
|
||
|
}
|
||
|
|
||
|
return pair()
|
||
|
}
|
||
|
|
||
|
func all() -> [Pair] {
|
||
|
var result: [Pair] = []
|
||
|
while let pair = self.pair() {
|
||
|
result.append(pair)
|
||
|
}
|
||
|
assert(isAtEnd)
|
||
|
return result
|
||
|
}
|
||
|
}
|
||
|
|
||
|
struct ExplodeContext {
|
||
|
var lastRegular: Regular? = nil
|
||
|
var addToNextNumber: Int? = nil
|
||
|
var didExplode = false
|
||
|
}
|
||
|
|
||
|
enum Action {
|
||
|
case goOn
|
||
|
case stop
|
||
|
case replace(Number)
|
||
|
}
|
||
|
|
||
|
protocol Number: AnyObject {
|
||
|
func toString() -> String
|
||
|
|
||
|
func visitExplode(depth: Int, context: inout ExplodeContext) -> Action
|
||
|
func visitSplit(didSplit: inout Bool) -> Action
|
||
|
|
||
|
func magnitude() -> Int
|
||
|
|
||
|
func copy() -> Number
|
||
|
}
|
||
|
|
||
|
extension Number where Self == Pair {
|
||
|
static func parse(_ string: String) -> Self {
|
||
|
Scanner(string: string).pair()!
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class Regular: Number {
|
||
|
var value: Int
|
||
|
|
||
|
init(value: Int) {
|
||
|
self.value = value
|
||
|
}
|
||
|
|
||
|
func toString() -> String {
|
||
|
"\(value)"
|
||
|
}
|
||
|
|
||
|
func visitExplode(depth: Int, context: inout ExplodeContext) -> Action {
|
||
|
if let add = context.addToNextNumber {
|
||
|
value += add
|
||
|
return .stop
|
||
|
}
|
||
|
|
||
|
context.lastRegular = self
|
||
|
return .goOn
|
||
|
}
|
||
|
|
||
|
func visitSplit(didSplit: inout Bool) -> Action {
|
||
|
if value >= 10 {
|
||
|
let first = value / 2
|
||
|
didSplit = true
|
||
|
return .replace(Pair(Regular(value: first), Regular(value: value - first)))
|
||
|
}
|
||
|
|
||
|
return .goOn
|
||
|
}
|
||
|
|
||
|
func magnitude() -> Int {
|
||
|
value
|
||
|
}
|
||
|
|
||
|
func copy() -> Number { Regular(value: value) }
|
||
|
|
||
|
}
|
||
|
|
||
|
class Pair: Number {
|
||
|
var left: Number
|
||
|
var right: Number
|
||
|
|
||
|
init(_ left: Number, _ right: Number) {
|
||
|
self.left = left
|
||
|
self.right = right
|
||
|
}
|
||
|
|
||
|
func toString() -> String {
|
||
|
"[\(left.toString()), \(right.toString())]"
|
||
|
}
|
||
|
|
||
|
func splitFirst() -> Bool {
|
||
|
var didSplit = false
|
||
|
_ = visitSplit(didSplit: &didSplit)
|
||
|
return didSplit
|
||
|
}
|
||
|
|
||
|
func explodeFirst() -> Bool {
|
||
|
var context = ExplodeContext()
|
||
|
_ = visitExplode(depth: 0, context: &context)
|
||
|
return context.didExplode
|
||
|
}
|
||
|
|
||
|
func reduce() -> Pair {
|
||
|
while true {
|
||
|
if explodeFirst() { continue }
|
||
|
if splitFirst() { continue }
|
||
|
break
|
||
|
}
|
||
|
|
||
|
return self
|
||
|
}
|
||
|
|
||
|
func visitExplode(depth: Int, context: inout ExplodeContext) -> Action {
|
||
|
if !context.didExplode && depth == 4 {
|
||
|
context.lastRegular?.value += (left as! Regular).value
|
||
|
context.addToNextNumber = (right as! Regular).value
|
||
|
context.didExplode = true
|
||
|
return .replace(Regular(value: 0))
|
||
|
}
|
||
|
|
||
|
|
||
|
switch left.visitExplode(depth: depth + 1, context: &context) {
|
||
|
case .goOn:
|
||
|
break
|
||
|
case .stop:
|
||
|
return .stop
|
||
|
case .replace(let number):
|
||
|
left = number
|
||
|
break
|
||
|
}
|
||
|
|
||
|
let result = right.visitExplode(depth: depth + 1, context: &context)
|
||
|
switch result {
|
||
|
case .goOn:
|
||
|
return .goOn
|
||
|
case .stop:
|
||
|
return .stop
|
||
|
case .replace(let number):
|
||
|
right = number
|
||
|
return .goOn
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
func visitSplit(didSplit: inout Bool) -> Action {
|
||
|
switch left.visitSplit(didSplit: &didSplit) {
|
||
|
case .goOn: break
|
||
|
case .stop: return .stop
|
||
|
case .replace(let new):
|
||
|
left = new
|
||
|
return .stop
|
||
|
}
|
||
|
|
||
|
switch right.visitSplit(didSplit: &didSplit) {
|
||
|
case .goOn: return .goOn
|
||
|
case .stop: return .stop
|
||
|
case .replace(let new):
|
||
|
right = new
|
||
|
return .stop
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func magnitude() -> Int {
|
||
|
3 * left.magnitude() + 2 * right.magnitude()
|
||
|
}
|
||
|
|
||
|
func copy() -> Number {
|
||
|
Pair(left.copy(), right.copy())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func +(lhs: Number, rhs: Number) -> Pair {
|
||
|
Pair(lhs.copy(), rhs.copy()).reduce()
|
||
|
}
|
||
|
|
||
|
let input = """
|
||
|
[4,[3,[9,[9,0]]]]
|
||
|
[[[7,6],[2,[2,5]]],[5,[[7,3],8]]]
|
||
|
[4,[4,6]]
|
||
|
[[0,[5,6]],[[[1,3],[2,7]],[[0,6],4]]]
|
||
|
[6,[[3,[6,0]],3]]
|
||
|
[[7,[9,[8,5]]],[6,7]]
|
||
|
[[[[2,6],1],2],[3,[8,4]]]
|
||
|
[4,[[[5,4],[2,7]],[[8,0],[2,3]]]]
|
||
|
[[[[4,3],2],[[3,6],[2,5]]],[[[3,7],8],0]]
|
||
|
[[[8,[0,7]],1],[[9,[3,9]],9]]
|
||
|
[[[[3,0],[1,3]],[[0,9],8]],[[[7,2],9],[[1,4],[3,5]]]]
|
||
|
[[[[9,6],[4,4]],[1,3]],[[4,3],[[6,4],[8,4]]]]
|
||
|
[[[1,2],[[7,6],[2,3]]],[[4,6],[4,2]]]
|
||
|
[[[4,8],[[5,8],1]],[2,3]]
|
||
|
[[[5,2],[3,[5,7]]],[[2,9],5]]
|
||
|
[[[6,[3,2]],[2,6]],[[8,[4,2]],[[5,2],7]]]
|
||
|
[[[[2,6],[0,1]],[7,[3,6]]],[[1,6],[[7,9],0]]]
|
||
|
[[[0,3],[8,1]],[[[9,0],3],[0,2]]]
|
||
|
[[8,[[7,1],[4,7]]],[[0,[1,3]],[8,2]]]
|
||
|
[[[[2,3],4],[[0,8],[9,0]]],[1,[[5,3],4]]]
|
||
|
[[[[7,2],2],[[1,3],[8,3]]],[4,[[7,9],[0,6]]]]
|
||
|
[[[[2,2],[3,4]],[[1,5],[4,3]]],[6,[[7,2],1]]]
|
||
|
[1,[[[5,7],0],[9,[8,8]]]]
|
||
|
[[[[9,2],[0,9]],[4,[7,8]]],[[4,8],[[1,8],[4,9]]]]
|
||
|
[[[[4,7],2],2],4]
|
||
|
[1,[[2,[4,2]],1]]
|
||
|
[[[[7,2],[3,8]],[0,[1,3]]],[[[4,4],[2,4]],[8,2]]]
|
||
|
[[[[1,0],[0,5]],2],[[9,[5,0]],[[1,6],5]]]
|
||
|
[4,[[[8,1],[1,4]],[7,[1,3]]]]
|
||
|
[[[6,[0,4]],[[4,6],[2,4]]],[9,[1,5]]]
|
||
|
[[[[3,6],[3,3]],1],[0,[[8,8],2]]]
|
||
|
[[7,[5,[2,6]]],[[[7,9],6],[0,[3,6]]]]
|
||
|
[[[[6,7],4],[[2,9],2]],3]
|
||
|
[[[7,[1,7]],[5,4]],[[[1,1],[0,1]],5]]
|
||
|
[[6,[[1,0],6]],[0,[6,[0,5]]]]
|
||
|
[[[[2,4],[4,6]],9],[4,[[8,0],7]]]
|
||
|
[[[[9,9],[5,7]],[9,[8,6]]],[[3,[2,3]],0]]
|
||
|
[[0,[1,[5,3]]],[3,[8,[3,4]]]]
|
||
|
[[[[4,3],8],[2,9]],[[1,[6,5]],[[5,7],2]]]
|
||
|
[[[0,[7,4]],[9,[9,6]]],[[8,[5,5]],[[6,4],1]]]
|
||
|
[[[[7,3],[7,9]],[8,[6,2]]],[[8,[4,5]],[[6,4],[6,7]]]]
|
||
|
[[7,[[9,0],[9,0]]],[[[0,8],2],[8,[8,3]]]]
|
||
|
[4,[7,[5,6]]]
|
||
|
[7,[[[3,8],8],3]]
|
||
|
[[[4,[6,6]],0],[9,0]]
|
||
|
[[[[7,4],8],8],[[0,1],[[0,0],[2,4]]]]
|
||
|
[7,[1,[[9,4],[3,6]]]]
|
||
|
[[[[2,8],9],[[8,6],[2,2]]],[[[5,1],9],[2,[0,7]]]]
|
||
|
[8,7]
|
||
|
[[[[0,8],4],[[9,9],[9,9]]],[[[4,3],[1,0]],[6,8]]]
|
||
|
[[[[8,3],[8,9]],1],[[4,[1,0]],[[4,0],[2,3]]]]
|
||
|
[[[[4,7],[1,3]],[6,9]],[[1,0],[[1,8],5]]]
|
||
|
[[2,[4,[6,5]]],[3,[[9,9],5]]]
|
||
|
[[[[7,6],4],9],[8,[4,5]]]
|
||
|
[[[0,[6,6]],[7,[8,9]]],[[[0,0],[3,4]],[4,[1,8]]]]
|
||
|
[[[9,[7,0]],[5,8]],[6,[[5,0],[0,6]]]]
|
||
|
[[[[4,0],[1,9]],[7,[3,6]]],[[2,[8,6]],[[2,8],[8,2]]]]
|
||
|
[[[9,6],8],[[[5,5],[4,8]],0]]
|
||
|
[[[[1,7],1],2],[[[6,8],3],[[3,3],5]]]
|
||
|
[3,[5,[[3,8],6]]]
|
||
|
[3,[[[9,6],[5,8]],[9,2]]]
|
||
|
[[6,1],[6,4]]
|
||
|
[[2,6],[[[1,2],2],8]]
|
||
|
[[[[1,7],[3,6]],[2,[0,2]]],[[3,0],9]]
|
||
|
[1,[[0,[4,9]],5]]
|
||
|
[[[[5,5],[5,2]],[0,[6,4]]],8]
|
||
|
[0,[7,[[6,9],[6,0]]]]
|
||
|
[[[[2,2],[4,7]],[[7,4],6]],[[0,[1,7]],[[3,2],6]]]
|
||
|
[[9,8],0]
|
||
|
[[[[5,4],[4,8]],2],[3,[8,9]]]
|
||
|
[[[[7,0],8],5],[2,6]]
|
||
|
[[[5,[0,8]],5],[[[5,0],[1,8]],[[0,2],7]]]
|
||
|
[[[[9,4],8],[[6,5],4]],[[5,[8,9]],[4,[0,4]]]]
|
||
|
[[[[3,6],7],[[9,3],7]],[7,[[8,3],9]]]
|
||
|
[[[[0,7],5],[[5,7],2]],[[2,[9,5]],[[7,7],[5,0]]]]
|
||
|
[[[[7,5],2],[8,6]],[[2,[6,2]],[5,[3,1]]]]
|
||
|
[[9,[9,1]],6]
|
||
|
[[[0,7],[[5,9],2]],3]
|
||
|
[[[9,3],[8,8]],[0,[4,5]]]
|
||
|
[[[[6,2],5],[4,[3,1]]],[9,[2,8]]]
|
||
|
[[[1,[9,4]],[[0,0],2]],[[1,[2,1]],[[7,8],[3,2]]]]
|
||
|
[[[[0,6],[8,9]],[[4,7],[5,6]]],[[[1,4],[8,7]],[4,6]]]
|
||
|
[[[[6,4],[1,5]],[0,8]],[[[9,7],[1,2]],[9,4]]]
|
||
|
[[[[4,5],[0,7]],[9,[1,8]]],[[[5,0],6],7]]
|
||
|
[[[0,[6,9]],[5,[5,6]]],7]
|
||
|
[[4,5],[[7,[6,5]],1]]
|
||
|
[[[7,9],[6,7]],[4,1]]
|
||
|
[[[[9,6],1],[[3,1],[9,7]]],[1,[7,1]]]
|
||
|
[[[0,[2,0]],5],[[8,[7,6]],[[7,3],4]]]
|
||
|
[[[6,[1,7]],[9,[2,7]]],3]
|
||
|
[[[6,[8,2]],5],[4,[[1,3],[5,1]]]]
|
||
|
[[[4,[3,3]],[4,[2,4]]],[5,4]]
|
||
|
[[[1,6],[4,[4,0]]],[[8,[2,2]],[[8,1],[4,7]]]]
|
||
|
[[2,0],[[2,1],[[4,8],[2,7]]]]
|
||
|
[9,[[8,4],0]]
|
||
|
[[1,6],[[5,[1,3]],[9,[0,9]]]]
|
||
|
[[[0,[3,5]],3],[[2,[8,0]],[[2,0],[4,3]]]]
|
||
|
[[[1,[1,9]],[9,[7,9]]],[[2,2],[[6,7],[0,7]]]]
|
||
|
[[[4,6],[[6,2],[0,9]]],[[1,0],[1,[6,7]]]]
|
||
|
[9,[[[0,1],4],[[9,3],3]]]
|
||
|
"""
|