From 62b7574d0db81362329fbfaef1be53de0f19547c Mon Sep 17 00:00:00 2001 From: Jack Kinsey Date: Fri, 13 Dec 2024 01:43:12 -0500 Subject: [PATCH] Complete day 12 part 2 --- src/day12.rs | 123 +++++++++++++++++++++++++++++++++++++++------------ 1 file changed, 94 insertions(+), 29 deletions(-) diff --git a/src/day12.rs b/src/day12.rs index a8fd499..cda9440 100644 --- a/src/day12.rs +++ b/src/day12.rs @@ -7,19 +7,19 @@ type Plot = char; struct Point(i32, i32); #[derive(Debug, Clone)] -struct Region { - plots: HashSet, - plant: char, +struct Region { + points: HashSet, + content: T, } -impl Region { +impl Region { fn perimeter(&self) -> u64 { - self.plots + self.points .iter() .map(|p| { 4 - p - .adjacent(i32::MAX.try_into().unwrap()) - .filter(|q| self.plots.contains(q)) + .adjacent_unchecked() + .filter(|q| self.points.contains(q)) .count() }) .sum::() @@ -28,12 +28,53 @@ impl Region { } fn area(&self) -> u64 { - self.plots.len().try_into().unwrap() + self.points.len().try_into().unwrap() + } + + fn sides(&self) -> u64 { + DIRS.iter() + .copied() + .map(|d| { + let ps: Vec<_> = self + .points + .iter() + .copied() + .map(move |p| p + d) + .filter(|p| !self.points.contains(p)) + .collect(); + let is: Vec<_> = ps.iter().copied().map(|Point(i, _)| i).collect(); + let js: Vec<_> = ps.iter().copied().map(|Point(_, j)| j).collect(); + let (imin, imax) = (is.iter().min(), is.iter().max()); + let (jmin, jmax) = (js.iter().min(), js.iter().max()); + let (&min, &max) = (imin.min(jmin).unwrap(), imax.max(jmax).unwrap()); + let n: usize = (max - min).try_into().unwrap(); + let mut map = Map::of(false, n + 1); + for p in ps { + map[p - Point(min, min)] = true; + } + let v: u64 = map + .regions() + .filter( + |&Region { + points: _, + content: c, + }| c, + ) + .count() + .try_into() + .unwrap(); + v + }) + .sum::() } fn price(&self) -> u64 { self.area() * self.perimeter() } + + fn discount_price(&self) -> u64 { + self.area() * self.sides() + } } #[rustfmt::skip] @@ -50,11 +91,13 @@ impl Point { } fn adjacent(self, bound: usize) -> impl Iterator { - DIRS.iter() - .copied() - .map(move |d| d + self) + self.adjacent_unchecked() .filter(move |p| p.in_bounds(bound)) } + + fn adjacent_unchecked(self) -> impl Iterator { + DIRS.iter().copied().map(move |d| self + d) + } } impl Add for Point { @@ -103,35 +146,37 @@ impl Map { } } -impl Map { - fn regions(&self) -> MapRegions { +impl Map { + fn regions(&self) -> MapRegions { MapRegions::new(self) } + + fn of(val: T, n: usize) -> Self { + Map(std::iter::repeat_with(|| vec![val; n]).take(n).collect()) + } } #[derive()] -struct MapRegions<'a> { - map: &'a Map, +struct MapRegions<'a, T: Eq + Copy> { + map: &'a Map, seen: Map, iter: Vec, bound: usize, } -impl<'a> MapRegions<'a> { - fn new(map: &'a Map) -> Self { +impl<'a, T: Eq + Copy> MapRegions<'a, T> { + fn new(map: &'a Map) -> Self { MapRegions { map, - seen: Map(std::iter::repeat_with(|| vec![false; map.0.len()]) - .take(map.0.len()) - .collect()), + seen: Map::of(false, map.0.len()), iter: map.points().collect(), bound: map.0.len(), } } } -impl Iterator for MapRegions<'_> { - type Item = Region; +impl Iterator for MapRegions<'_, T> { + type Item = Region; fn next(&mut self) -> Option { let mut p = self.iter.pop()?; @@ -154,7 +199,10 @@ impl Iterator for MapRegions<'_> { })); points.extend(iterators.drain(..).flatten()); } - Some(Region { plots, plant: c }) + Some(Region { + points: plots, + content: c, + }) } } @@ -167,7 +215,10 @@ pub fn part1(input: &str) -> u64 { } pub fn part2(input: &str) -> u64 { - 0 + parse(input) + .regions() + .map(|r| r.discount_price()) + .sum::() } #[cfg(test)] @@ -226,7 +277,12 @@ mod test { assert_eq!( parse(TINY_INPUT) .regions() - .map(|Region { plots: _, plant: c }| c) + .map( + |Region { + points: _, + content: c, + }| c + ) .collect::>(), ['C', 'E', 'B', 'D', 'A'], ) @@ -241,14 +297,23 @@ mod test { } #[test] - #[ignore] + fn test_discount_price() { + assert_eq!( + parse(INPUT_STR) + .regions() + .map(|r| r.discount_price()) + .sum::(), + 1206 + ) + } + + #[test] fn test_part1() { - assert_eq!(part1(&crate::input(0).unwrap()), 0) + assert_eq!(part1(&crate::input(12).unwrap()), 1433460) } #[test] - #[ignore] fn test_part2() { - assert_eq!(part2(&crate::input(0).unwrap()), 0) + assert_eq!(part2(&crate::input(12).unwrap()), 855082) } } -- 2.38.5