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
|
use std::io;
fn main() -> io::Result<()> {
let eqns = io::stdin()
.lines()
.flatten()
.collect::<Vec<_>>()
.iter()
.map(|line| line.split_once(": ").unwrap())
.map(|(ans, nums)| {
(
ans.parse::<u64>().unwrap(),
nums.split_whitespace()
.map(|s| s.parse::<u64>().unwrap())
.collect::<Vec<_>>(),
)
})
.collect::<Vec<_>>();
let silver: u64 = eqns
.iter()
.filter(|(ans, nums)| calibration(ans, nums, false))
.map(|(ans, _)| ans)
.sum();
let gold: u64 = eqns
.iter()
.filter(|(ans, nums)| calibration(ans, nums, true))
.map(|(ans, _)| ans)
.sum();
println!("silver: {silver}");
println!("gold: {gold}");
return Ok(());
}
fn concat(x: &u64, y: &u64) -> u64 {
x * (10u64.pow(y.ilog10() + 1)) + y
}
fn calibration(want: &u64, nums: &[u64], gold: bool) -> bool {
match nums {
[] => unreachable!(),
[x] => want == x,
[x, y, xs @ ..] => {
if x > want {
return false;
}
calibration(want, &[&[x + y], xs].concat(), gold) ||
calibration(want, &[&[x * y], xs].concat(), gold) ||
(gold && calibration(want, &[&[concat(x, y)], xs].concat(), gold) )
}
}
}
|