-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathday_05.rs
172 lines (146 loc) · 4.61 KB
/
day_05.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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
use std::collections::{btree_map::Entry, BTreeMap, VecDeque};
// Day 5 - Supply Stacks
//
// Couldn't get a good grip on parsing the stacks that early in the morning. It was
// easier to do it manually since it wasn't that many. Came back and created a parser for it.
//
// Refactored using a BTreeMap to store the stacks. It doesn't make a difference
// in performance, but it's simple to use.
//
// Took inspiration from @ankjevel and used a VecDeque instead of a Vec for the stacks.
// The solution became both simpler and faster. Really beautiful.
pub struct Instruction {
to: usize,
from: usize,
moves: usize,
}
// Instructions look like this:
// move 5 from 1 to 2
//
// Split it by spaces and pick out the numbers.
impl Instruction {
fn new(input: &str) -> Instruction {
let parts: Vec<&str> = input.split_whitespace().collect();
Instruction {
to: parts[5].parse::<usize>().unwrap() - 1,
from: parts[3].parse::<usize>().unwrap() - 1,
moves: parts[1].parse().unwrap(),
}
}
}
type Stacks = BTreeMap<usize, VecDeque<String>>;
type Instructions = Vec<Instruction>;
type Input = (Stacks, Instructions);
fn first_in_stacks(stacks: Stacks) -> String {
stacks
.values()
.map(|v| v.front().unwrap().to_string())
.collect::<Vec<String>>()
.join("")
}
#[aoc_generator(day5)]
pub fn input_generator(input: &str) -> Input {
let mut stacks: Stacks = BTreeMap::new();
let stacks_and_instructions: Vec<&str> = input.split("\n\n").collect();
// Parse the stacks. Remove the last line of column numbers.
// The example data is:
//
// [D]
// [N] [C]
// [Z] [M] [P]
// 1 2 3
for row in stacks_and_instructions[0]
.lines()
.collect::<Vec<_>>()
.split_last()
.unwrap()
.1
{
for (i, column) in row.chars().collect::<Vec<_>>().chunks(4).enumerate() {
// Find the name of the crate
let value: String = column
.iter()
.map(|s| s.to_string().trim().replace(['[', ']'], ""))
.collect::<Vec<String>>()
.join("");
// Skip empty columns
if value.is_empty() {
continue;
}
// Add to or create the stack
match stacks.entry(i) {
Entry::Vacant(e) => {
e.insert(VecDeque::from_iter([value]));
}
Entry::Occupied(mut e) => {
e.get_mut().push_back(value);
}
}
}
}
// Parse the instructions
let instructions: Vec<Instruction> = stacks_and_instructions[1]
.lines()
.filter(|s| !s.is_empty())
.map(Instruction::new)
.collect();
(stacks, instructions)
}
/* Part One
*
* Move each crate to the correct stack. One at a time.
*/
#[aoc(day5, part1)]
pub fn solve_part_01((stacks, instructions): &Input) -> String {
let mut stacks = stacks.clone();
for Instruction { moves, from, to } in instructions {
// Move the crates, one at a time
for _ in 0..*moves {
// Remove the top crate from the source stack
let crate_in_stack = stacks.get_mut(from).unwrap().pop_front().unwrap();
// Add it to the destination stack
stacks.get_mut(to).unwrap().push_front(crate_in_stack);
}
}
// Find the first crate in each stack
first_in_stacks(stacks)
}
/* Part Two
*
* Move each crate to the correct stack. All at once.
*/
#[aoc(day5, part2)]
pub fn solve_part_02((stacks, instructions): &Input) -> String {
let mut stacks = stacks.clone();
for Instruction { moves, from, to } in instructions {
// Remove all crates that should be moved from the source stack.
// Reverse the order since we're draining from the front.
let move_stack: VecDeque<_> = stacks.get_mut(from).unwrap().drain(..moves).rev().collect();
// Add the crates to the destination stack
for crate_in_stack in move_stack {
stacks.get_mut(to).unwrap().push_front(crate_in_stack);
}
}
// Find the first crate in each stack
first_in_stacks(stacks)
}
#[cfg(test)]
mod tests {
use super::*;
const SAMPLE: &str = " [D]
[N] [C]
[Z] [M] [P]
1 2 3
move 1 from 2 to 1
move 3 from 1 to 3
move 2 from 2 to 1
move 1 from 1 to 2";
#[test]
fn sample_01() {
assert_eq!(solve_part_01(&input_generator(SAMPLE)), "CMZ")
}
#[test]
fn sample_02() {
assert_eq!(solve_part_02(&input_generator(SAMPLE)), "MCD")
}
}