forked from maneatingape/advent-of-code-rust
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday22.rs
87 lines (70 loc) · 2.48 KB
/
day22.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
82
83
84
85
86
87
use crate::util::iter::*;
use crate::util::parse::*;
type Input = (usize, usize);
pub fn parse(input: &str) -> Input {
let mut bricks: Vec<_> = input.iter_unsigned::<usize>().chunk::<6>().collect();
let mut heights = [0; 100];
let mut indices = [usize::MAX; 100];
let mut safe = vec![true; bricks.len()];
let mut dominator: Vec<(usize, usize)> = Vec::with_capacity(bricks.len());
// Sort ascending by lowest z coordinate.
bricks.sort_unstable_by_key(|b| b[2]);
for (i, &[x1, y1, z1, x2, y2, z2]) in bricks.iter().enumerate() {
let start = 10 * y1 + x1;
let end = 10 * y2 + x2;
let step = if y2 > y1 { 10 } else { 1 };
let height = z2 - z1 + 1;
let mut top = 0;
let mut previous = usize::MAX;
let mut underneath = 0;
let mut parent = 0;
let mut depth = 0;
for j in (start..=end).step_by(step) {
top = top.max(heights[j]);
}
for j in (start..=end).step_by(step) {
if heights[j] == top {
let index = indices[j];
if index != previous {
previous = index;
underneath += 1;
if underneath == 1 {
(parent, depth) = dominator[previous];
} else {
// Find common ancestor
let (mut a, mut b) = (parent, depth);
let (mut x, mut y) = dominator[previous];
while b > y {
(a, b) = dominator[a];
}
while y > b {
(x, y) = dominator[x];
}
while a != x {
(a, b) = dominator[a];
(x, _) = dominator[x];
}
(parent, depth) = (a, b);
}
}
}
heights[j] = top + height;
indices[j] = i;
}
if underneath == 1 {
safe[previous] = false;
parent = previous;
depth = dominator[previous].1 + 1;
}
dominator.push((parent, depth));
}
let part_one = safe.iter().filter(|&&b| b).count();
let part_two = dominator.iter().map(|(_, d)| d).sum();
(part_one, part_two)
}
pub fn part1(input: &Input) -> usize {
input.0
}
pub fn part2(input: &Input) -> usize {
input.1
}