aoc23/src/bin/day9.rs

78 lines
1.6 KiB
Rust
Raw Normal View History

2023-12-09 12:42:43 +01:00
#![allow(clippy::needless_range_loop)]
#![forbid(elided_lifetimes_in_paths, unsafe_code)]
use aoc23::read;
2023-12-09 12:47:59 +01:00
use chumsky::prelude::*;
2023-12-09 12:42:43 +01:00
struct History(Vec<i64>);
impl History {
fn parser() -> impl Parser<char, Self, Error = Simple<char>> {
one_of(['-', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9'])
.repeated()
.at_least(1)
.separated_by(just(" "))
.map(|nums: Vec<Vec<char>>| {
Self(
nums.into_iter()
.map(|num| num.into_iter().collect::<String>().parse().unwrap())
.collect()
)
})
}
}
fn parser() -> impl Parser<char, Vec<History>, Error = Simple<char>> {
History::parser()
.then_ignore(just("\n"))
.repeated()
.then_ignore(end())
}
impl History {
fn diff(&self) -> Self {
2023-12-09 12:47:59 +01:00
let mut diffs = Vec::with_capacity(self.0.len() - 1);
2023-12-09 12:42:43 +01:00
if self.0.len() > 1 {
for i in 1 .. self.0.len() {
diffs.push(self.0[i] - self.0[i - 1]);
}
}
Self(diffs)
}
2023-12-09 12:47:59 +01:00
fn extend_last(&self) -> i64 {
2023-12-09 12:42:43 +01:00
if self.0.iter().any(|num| *num != 0) {
2023-12-09 12:47:59 +01:00
let diff = self.diff();
let last = diff.extend_last();
self.0.last().unwrap() + last
2023-12-09 12:42:43 +01:00
} else {
2023-12-09 12:47:59 +01:00
0
}
}
fn extend_first(&self) -> i64 {
if self.0.iter().any(|num| *num != 0) {
let diff = self.diff();
let first = diff.extend_first();
self.0.first().unwrap() - first
} else {
0
2023-12-09 12:42:43 +01:00
}
}
}
fn main() -> anyhow::Result<()> {
let histories = read("inputs/day9.txt", parser())?;
2023-12-09 12:47:59 +01:00
let mut sum_last = 0;
let mut sum_first = 0;
for h in histories {
sum_last += h.extend_last();
sum_first += h.extend_first();
2023-12-09 12:42:43 +01:00
}
2023-12-09 12:47:59 +01:00
println!("{sum_last}");
println!("{sum_first}");
2023-12-09 12:42:43 +01:00
Ok(())
}