1
0
Fork 0
This commit is contained in:
Lars Martens 2024-12-10 07:37:53 +01:00
parent e7b127ea01
commit 79b809c3b7
Signed by: haselkern
GPG key ID: B5CF1F363C179AD4

111
src/bin/10.rs Normal file
View file

@ -0,0 +1,111 @@
use aoc::*;
use glam::IVec2;
use std::collections::{HashMap, HashSet};
use std::iter;
const INPUT: &str = include_str!("../../input/10");
fn main() {
assert_example!(part1, "10-test", 36);
println!("Part 1: {}", part1(INPUT));
assert_example!(part2, "10-test", 81);
println!("Part 2: {}", part2(INPUT));
}
fn part1(input: &str) -> usize {
let map = Map::parse(input);
map.trailheads().map(|t| map.reachable(t).len()).sum()
}
fn part2(input: &str) -> usize {
let map = Map::parse(input);
map.trailheads().map(|t| map.distinct_trails(t)).sum()
}
fn directions() -> [IVec2; 4] {
[
IVec2::new(1, 0),
IVec2::new(-1, 0),
IVec2::new(0, 1),
IVec2::new(0, -1),
]
}
#[derive(Debug)]
struct Map {
height: HashMap<IVec2, i64>,
}
impl Map {
fn reachable(&self, from: IVec2) -> HashSet<IVec2> {
let from_height = match self.get_height(from) {
Some(9) => return [from].into(),
None => return [].into(),
Some(h) => h,
};
self.climb_up(from, from_height)
.into_iter()
.map(|pos| self.reachable(pos))
.fold(HashSet::new(), |mut acc, other| {
acc.extend(other);
acc
})
}
fn distinct_trails(&self, from: IVec2) -> usize {
let from_height = match self.get_height(from) {
Some(9) => return 1,
None => return 0,
Some(h) => h,
};
self.climb_up(from, from_height)
.into_iter()
.map(|pos| self.distinct_trails(pos))
.sum()
}
fn climb_up(&self, from: IVec2, from_height: i64) -> Vec<IVec2> {
directions()
.into_iter()
.filter_map(|dir| {
let pos = from + dir;
self.get_height(pos).map(|height| (pos, height))
})
.filter_map(|(pos, h)| {
if h == from_height + 1 {
Some(pos)
} else {
None
}
})
.collect()
}
fn get_height(&self, at: IVec2) -> Option<i64> {
self.height.get(&at).copied()
}
fn trailheads(&self) -> impl Iterator<Item = IVec2> + use<'_> {
self.height.iter().filter_map(|(&pos, &h)| match h {
0 => Some(pos),
_ => None,
})
}
fn parse(input: &str) -> Self {
let height = input
.lines()
.enumerate()
.flat_map(|(y, line)| line.chars().enumerate().zip(iter::repeat(y)))
.map(|((x, c), y)| {
let pos = IVec2::new(x as i32, y as i32);
let height = c.to_digit(10).unwrap() as i64;
(pos, height)
})
.collect();
Self { height }
}
}