-
-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathmain.rs
65 lines (59 loc) · 1.96 KB
/
main.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
use std::collections::BTreeMap;
const EDGES: usize = 12;
pub fn main() {
let mut uc = vec![];
let mut id: BTreeMap<&str, u8> = BTreeMap::from([("start", 1), ("end", 0)]);
let mut map: BTreeMap<u8, Vec<u8>> = BTreeMap::new();
include_str!("../input.txt").lines().for_each(|l| {
let mut idx = |a| {
let idx = id.len() as u8;
*id.entry(a).or_insert_with(|| {
(a.as_bytes()[0] <= b'Z').then(|| uc.push(idx));
idx
})
};
let mut branch = |a, b| {
let entry = map.entry(a).or_insert_with(|| Vec::with_capacity(6));
(b != 0).then(|| entry.push(b));
};
let (a, b) = l.split_once('-').unwrap();
let (a, b) = (idx(a), idx(b));
branch(a, b);
branch(b, a);
});
let map = map
.keys()
.filter(|b| !uc.contains(b))
.map(|&b| {
(
b,
map[&b].iter().fold([0; EDGES], |mut chld, b| {
if uc.contains(b) {
map[b].iter().for_each(|b| chld[*b as usize] += 1);
} else {
chld[*b as usize] += 1;
}
chld
}),
)
})
.collect::<BTreeMap<_, _>>();
let mut todo: Vec<(u8, u8, usize)> = vec![(0, 1, 1)];
let (mut to, mut count) = ([1; EDGES], 0);
while let Some((a, b, s)) = todo.pop() {
to[b as usize] = a;
count += map[&a]
.iter()
.enumerate()
.filter(|&(_, routes)| *routes > 0)
.fold(0, |acc, (c, _)| match c {
1 => acc + s * map[&a][c],
v => {
let visit = v != 0 && to[2..=b as usize].contains(&(v as u8));
(!visit).then(|| todo.push((v as u8, b + 1, s * map[&a][v])));
acc
}
});
}
println!("{}", count);
}