AoC/day12.swift

103 lines
2.7 KiB
Swift
Raw Normal View History

2021-12-12 12:11:27 +01:00
@main
struct Day12: Puzzle {
func run() {
let rooms = input.reduce(into: Set<String>()) { partialResult, pair in
partialResult.insert(pair.0)
partialResult.insert(pair.1)
}.sorted()
let small = Set(rooms.enumerated().compactMap { $0.element.first!.isLowercase ? $0.offset : nil })
let matrix = buildMatrix(rooms, small: small)
let startIndex = rooms.firstIndex(of: "start")!
let endIndex = rooms.firstIndex(of: "end")!
let paths = matrix.findPaths(from: startIndex, to: endIndex, continuing: Path())
print("total paths", paths.count)
}
struct Path {
var rooms: [Int] = []
2021-12-12 12:14:49 +01:00
func appending(_ room: Int) -> Path {
Path(rooms: rooms + [room])
}
2021-12-12 12:11:27 +01:00
}
struct Matrix {
var data: [Bool]
let size: Int
let small: Set<Int>
func neighbors(of index: Int) -> [Int] {
let slice = data[index * size ..< (index + 1) * size]
return slice.enumerated().compactMap { $0.element ? $0.offset : nil }
}
func findPaths(from: Int, to: Int, continuing: Path) -> [Path] {
guard from != to else {
return [continuing]
}
2021-12-12 12:14:49 +01:00
let current = continuing.appending(from)
2021-12-12 12:11:27 +01:00
var result: [Path] = []
for next in neighbors(of: from) {
if small.contains(next) && current.rooms.contains(next) {
continue
}
let nextPaths = findPaths(from: next, to: to, continuing: current)
result.append(contentsOf: nextPaths)
}
return result
}
}
func buildMatrix(_ rooms: [String], small: Set<Int>) -> Matrix {
let count = rooms.count
var matrix = [Bool](repeating: false, count: count * count)
for (from, to) in input {
let fromIndex = rooms.firstIndex(of: from)!
let toIndex = rooms.firstIndex(of: to)!
matrix[fromIndex + count * toIndex] = true
matrix[toIndex + count * fromIndex] = true
}
return Matrix(data: matrix, size: count, small: small)
}
let input: [(String, String)] = [
("rf", "RL"),
("rf", "wz"),
("wz", "RL"),
("AV", "mh"),
("end", "wz"),
("end", "dm"),
("wz", "gy"),
("wz", "dm"),
("cg", "AV"),
("rf", "AV"),
("rf", "gy"),
("end", "mh"),
("cg", "gy"),
("cg", "RL"),
("gy", "RL"),
("VI", "gy"),
("AV", "gy"),
("dm", "rf"),
("start", "cg"),
("start", "RL"),
("rf", "mh"),
("AV", "start"),
("qk", "mh"),
("wz", "mh"),
]
}