summaryrefslogtreecommitdiff
path: root/2024/07/rust/src/main.rs
blob: df586b1db279c91aceec0b636f4567d4c97d8f8e (plain)
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
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().map(crate::silver).sum();
    let gold: u64 = eqns.iter().map(crate::gold).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, x: &u64, y: &u64, rest: &[u64], gold: bool) -> bool {
    if *x > want {
        return false;
    }
    match rest {
        [] => want == x + y || want == x * y || (gold && want == concat(x, y)),
        [z, rest @ ..] => {
            calibration(want, &(x + y), z, rest, gold)
                || calibration(want, &(x * y), z, rest, gold)
                || (gold && calibration(want, &concat(x, y), z, rest, gold))
        }
    }
}

fn silver((want, nums): &(u64, Vec<u64>)) -> u64 {
    match nums.as_slice() {
        [x, y, rest @ ..] => {
            if calibration(*want, x, y, rest, false) {
                *want
            } else {
                0
            }
        }
        _ => 0,
    }
}

fn gold((want, nums): &(u64, Vec<u64>)) -> u64 {
    match nums.as_slice() {
        [x, y, rest @ ..] => {
            if calibration(*want, x, y, rest, true) {
                *want
            } else {
                0
            }
        }
        _ => 0,
    }
}