-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathday09.rs
74 lines (63 loc) · 2.12 KB
/
day09.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
//! # Smoke Basin
//!
//! Part 2 is the classic [flood fill](https://en.wikipedia.org/wiki/Flood_fill) algorithm with a
//! twist to return the size of the filled area. This algorithm can be implemented either as a
//! [DFS](https://en.wikipedia.org/wiki/Depth-first_search) using recursion or as a
//! [BFS](https://en.wikipedia.org/wiki/Breadth-first_search) using an auxilary data structure
//! such as a [`VecDeque`].
//!
//! This solution uses a DFS approach as it's faster and Rust's stack size limit seems enough
//! to accommodate the maximum basin size. 2 dimensional grids are common in Advent of Code
//! problems so we use our utility [`Grid`] and [`Point`] modules.
//!
//! [`VecDeque`]: std::collections::VecDeque
//! [`Grid`]: crate::util::grid
//! [`Point`]: crate::util::point
use crate::util::grid::*;
use crate::util::parse::*;
use crate::util::point::*;
pub fn parse(input: &str) -> Grid<u8> {
Grid::parse(input)
}
pub fn part1(grid: &Grid<u8>) -> u32 {
let mut risk_levels = 0;
for x in 0..grid.width {
for y in 0..grid.height {
let point = Point::new(x, y);
let cur = grid[point];
let low_point = ORTHOGONAL
.iter()
.map(|&n| point + n)
.filter(|&n| grid.contains(n))
.all(|n| grid[n] > cur);
if low_point {
risk_levels += 1 + cur.to_decimal() as u32;
}
}
}
risk_levels
}
pub fn part2(input: &Grid<u8>) -> u32 {
let mut grid = input.clone();
let mut basins = Vec::new();
for x in 0..grid.width {
for y in 0..grid.height {
let next = Point::new(x, y);
if grid[next] < b'9' {
basins.push(flood_fill(&mut grid, next));
}
}
}
basins.sort_unstable();
basins.iter().rev().take(3).product()
}
fn flood_fill(grid: &mut Grid<u8>, point: Point) -> u32 {
grid[point] = b'9';
let mut size = 1;
for next in ORTHOGONAL.iter().map(|&n| point + n) {
if grid.contains(next) && grid[next] < b'9' {
size += flood_fill(grid, next);
}
}
size
}