summaryrefslogtreecommitdiff
path: root/src/day_7.rs
blob: e14512a5d92657cf30e89b83fd95bce40e1a166c (plain) (blame)
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
use std::cmp;

fn abs_diff(a: u32, b: u32) -> u32 { cmp::max(a, b) - cmp::min(a, b) }

fn fuel_required_constant(positions: &[u32], destination: u32) -> u32
{
    positions.iter().fold(0, |fuel_total, pos| {
        fuel_total + abs_diff(*pos, destination)
    })
}

fn fuel_required_linear(positions: &[u32], destination: u32) -> u32
{
    let fuel_required = |pos, dest| {
        let distance = abs_diff(pos, dest);
        // Get total fuel cost by adding all integers until distance on top of each
        // other
        (distance * (distance + 1)) / 2
    };

    positions.iter().fold(0, |fuel_total, pos| {
        fuel_total + fuel_required(*pos, destination)
    })
}

fn part1(positions: &[u32])
{
    // Stupid approach: Try all positions out
    let mut min_required = fuel_required_constant(positions, 0);
    for pos in 1..positions.len() {
        min_required = cmp::min(fuel_required_constant(positions, pos as u32), min_required);
    }

    println!("Solution to part 1: {}", min_required);
}

fn part2(positions: &[u32])
{
    // Stupid approach: Try all positions out
    let mut min_required = fuel_required_linear(positions, 0);
    for pos in 1..positions.len() {
        min_required = cmp::min(fuel_required_linear(positions, pos as u32), min_required);
    }

    println!("Solution to part 2: {}", min_required);
}

pub fn run(input: Vec<String>)
{
    let positions: Vec<u32> = input
        .iter()
        .map(|s| s.split(',').map(|s| s.parse().unwrap()))
        .flatten()
        .collect();

    part1(&positions);
    part2(&positions);
}