summaryrefslogtreecommitdiff
path: root/2024/05/rust/src/main.rs
blob: bf72727370238d7c70f13580be52f1c5ff7bec63 (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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#![feature(slice_split_once)]

use std::cmp::Ordering;
use std::collections::HashMap;
use std::io;

fn main() -> io::Result<()> {
    let lines = io::stdin().lines().flatten().collect::<Vec<_>>();
    let (rules, updates) = lines.split_once(String::is_empty).unwrap();
    let manual = rules.iter().collect::<Manual>();

    let updates = updates
        .iter()
        .map(|up| {
            up.split(',')
                .map(|p| Page {
                    number: p.parse::<u64>().unwrap(),
                    manual: &manual,
                })
                .collect::<_>()
        })
        .collect::<Vec<Vec<_>>>();

    let mut silver = 0;
    let mut gold = 0;
    for mut up in updates {
        let mid = up.len() / 2;
        if up.is_sorted() {
            silver += up[mid].number;
        } else {
            up.sort_unstable();
            gold += up[mid].number;
        }
    }

    println!("silver: {silver}");
    println!("gold: {gold}");

    return Ok(());
}
#[derive(Debug)]
struct Manual {
    order: HashMap<u64, Vec<u64>>,
}

impl<'a> FromIterator<&'a String> for Manual {
    fn from_iter<T: IntoIterator<Item = &'a String>>(rules: T) -> Self {
        let mut order = HashMap::<u64, Vec<u64>>::new();
        for rule in rules {
            let (pre, post) = rule.split_once('|').unwrap();
            order
                .entry(pre.parse().unwrap())
                .or_default()
                .push(post.parse().unwrap());
        }
        Manual { order }
    }
}

#[derive(Clone, Copy)]
struct Page<'a> {
    number: u64,
    manual: &'a Manual,
}

impl PartialEq for Page<'_> {
    fn eq(&self, other: &Self) -> bool {
        self.number == other.number
    }
}
impl Eq for Page<'_> {}

impl PartialOrd for Page<'_> {
    fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
        if self.manual.order.get(&self.number)?.contains(&other.number) {
            Some(Ordering::Less)
        } else {
            Some(Ordering::Greater)
        }
    }
}

impl Ord for Page<'_> {
    fn cmp(&self, other: &Self) -> Ordering {
        self.partial_cmp(other).unwrap()
    }
}