-
Notifications
You must be signed in to change notification settings - Fork 15
/
day02.rs
81 lines (65 loc) · 2.1 KB
/
day02.rs
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
//! # Red-Nosed Reports
//!
//! Computes boths parts simultaneously. Each pair of levels is converted into deltas of
//! either +1, -1 or 0. For example:
//!
//! * `1 3 6 7 9` => `+1 +1 +1 +1`
//! * `9 7 6 2 1` => `-1 -1 0 -1`
//!
//! If the sum of all delta equals ±4, then we know that all levels are increasing or
//! decreasing. Any other value indicates either mixed up and down transitions or levels that
//! are too far apart.
//!
//! For part two we remove each pair of deltas (or single delta at each end) then replace with
//! the sum of the delta from the new neighbors on either side.
use crate::util::parse::*;
type Input = (u32, u32);
/// Minimize allocation to only a single `vec` reused for each report.
pub fn parse(input: &str) -> Input {
let mut report = Vec::new();
let mut part_one = 0;
let mut part_two = 0;
for line in input.lines() {
report.extend(line.iter_signed::<i32>());
let (p1, p2) = check(&report);
part_one += p1;
part_two += p2;
report.clear();
}
(part_one, part_two)
}
pub fn part1(input: &Input) -> u32 {
input.0
}
pub fn part2(input: &Input) -> u32 {
input.1
}
fn check(report: &[i32]) -> (u32, u32) {
let size = report.len();
let score: i32 = (1..size).map(|i| delta(report[i - 1], report[i])).sum();
if score.abs() == (size - 1) as i32 {
return (1, 1);
}
for i in 0..size {
let mut score = score;
// Snip out each level and replace with new level computed from neighbors to either side.
if i > 0 {
score -= delta(report[i - 1], report[i]);
}
if i < size - 1 {
score -= delta(report[i], report[i + 1]);
}
if i > 0 && i < size - 1 {
score += delta(report[i - 1], report[i + 1]);
}
if score.abs() == (size - 2) as i32 {
return (0, 1);
}
}
(0, 0)
}
/// Convert each pair of levels to either +1 for increase, -1 for decrease or 0 for invalid range.
fn delta(a: i32, b: i32) -> i32 {
let diff = b - a;
if diff.abs() <= 3 { diff.signum() } else { 0 }
}