summaryrefslogtreecommitdiff
path: root/2017/src/bin/day_24.rs
blob: eb7fddd9c0027ac0fefa9a08d54cd35016c73457 (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
extern crate advent_of_code_2017;
use advent_of_code_2017::*;

fn main() {
    let args = AdventArgs::init();
    let components: Vec<Component> = args.input.iter()
        .map(|line| {
            let mut split = line.split('/');
            Component {
                a: split.next().unwrap().parse().unwrap(),
                b: split.next().unwrap().parse().unwrap()
            }
        })
        .collect();

    if args.part == 1 {
        let strongest = build_strongest(0, components);
        println!("{}", strongest);
    } else {
        let (strongest, longest) = build_longest(0, components);
        println!("length: {}, strength: {}", longest, strongest);
    }
}

fn build_strongest(start: u32, components: Vec<Component>) -> u32 {
    components.iter().enumerate()
        .filter(|&(_, c)| c.a == start || c.b == start)
        .map(|(i, c)| {
            let end = if c.a == start { c.b } else { c.a };
            let mut subset = components.clone();
            subset.remove(i);
            c.strength() + build_strongest(end, subset)
        }).max().unwrap_or(0)
}

fn build_longest(start: u32, components: Vec<Component>) -> (u32, u32) {
    components.iter().enumerate()
        .filter(|&(_, c)| c.a == start || c.b == start)
        .map(|(i, c)| {
            let end = if c.a == start { c.b } else { c.a };
            let mut subset = components.clone();
            subset.remove(i);
            let (s, l) = build_longest(end, subset);
            (c.strength() + s, 1 + l)
        }).max_by(|&(s1, l1), &(s2, l2)| {
            l1.cmp(&l2).then(s1.cmp(&s2))
        }).unwrap_or((0, 0))
}

#[derive(Debug, Clone)]
struct Component {
    a: u32,
    b: u32
}

impl Component {
    fn strength(&self) -> u32 {
        self.a + self.b
    }
}