-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmod.rs
68 lines (55 loc) · 2.21 KB
/
mod.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
use std::collections::HashMap;
pub fn solve(input: String) {
let parts = input.split(',');
let mut boxes = vec![HashMap::<Vec<u8>,(usize,u8)>::new(); 256];
let mut max_order = vec![0; 256];
let mut sum_one = 0usize;
for part in parts {
let mut full_hash = 0u8;
let mut label_hash = 0u8;
let mut label = Vec::new();
for c in part.as_bytes() {
match c {
b'a'..=b'z' => {
// label
label_hash = label_hash.wrapping_add(*c);
label_hash = label_hash.wrapping_mul(17);
label.push(*c);
},
b'0'..=b'9' => {
// data => we expect a = to be before that, but never actually verify that - taking shortcuts here
let lens = *c - b'0';
// this way we retain the ordering independently of removals since a new one will always be at the end
max_order[label_hash as usize] += 1;
boxes[label_hash as usize].entry(label.clone()).and_modify(|b| {
b.1 = lens
}).or_insert((max_order[label_hash as usize], lens));
},
b'-' => {
// remove lens
boxes[label_hash as usize].remove(&label);
// we skip re-ordering of the other lenses and do the ordering in the end
},
b'\n' => {
// ignore new lines
continue;
}
// skip =
_ => {},
}
full_hash = full_hash.wrapping_add(*c);
full_hash = full_hash.wrapping_mul(17);
}
sum_one += full_hash as usize;
}
let mut sum_two = 0;
for (box_no, box_map) in boxes.iter().enumerate() {
let mut box_vec: Vec<_> = box_map.iter().map(|(_, data)| data).collect();
// order according to position
box_vec.sort_by_key(|x| x.0);
for (slot, (_, lens)) in box_vec.iter().enumerate() {
sum_two += (box_no + 1) * (slot + 1) * (*lens as usize);
}
}
aoc::print_solution(&[sum_one, sum_two]);
}