-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathday04.rs
120 lines (106 loc) · 3.21 KB
/
day04.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
use crate::data::load;
use linked_hash_set::LinkedHashSet;
use std::collections::HashMap;
use thiserror::Error;
#[derive(Error, Debug, PartialEq, Eq)]
pub enum PuzzleErr {
#[error("Could not parse card: '{}'.", .0)]
CardParsingError(String),
#[error("Could not parse numbers: '{}'.", .0)]
NumParsingError(String),
}
#[derive(Debug, Clone)]
struct Card {
id: u32,
win_nums: LinkedHashSet<u32>,
obs_nums: LinkedHashSet<u32>,
}
fn parse_nums(nums: &str) -> Result<LinkedHashSet<u32>, PuzzleErr> {
nums.split(' ')
.filter(|c| !c.is_empty())
.map(|x| {
x.trim()
.parse::<u32>()
.or(Err(PuzzleErr::NumParsingError(nums.to_string())))
})
.collect()
}
impl Card {
fn from_input(data: &str, id: u32) -> Result<Self, PuzzleErr> {
let split_data = data.split(':').collect::<Vec<_>>()[1]
.trim()
.split('|')
.collect::<Vec<_>>();
if split_data.len() != 2 {
return Err(PuzzleErr::CardParsingError(data.to_string()));
}
Ok(Card {
id,
win_nums: parse_nums(split_data[0])?,
obs_nums: parse_nums(split_data[1])?,
})
}
fn score(&self) -> u32 {
match self.n_hits() {
0 => 0,
1 => 1,
x => 2_u32.pow(x - 1),
}
}
fn n_hits(&self) -> u32 {
self.win_nums
.intersection(&self.obs_nums)
.collect::<Vec<_>>()
.len() as u32
}
}
fn parse_input(input: &str) -> Result<Vec<Card>, PuzzleErr> {
input
.trim()
.lines()
.enumerate()
.map(|(i, t)| Card::from_input(t, i as u32 + 1))
.collect::<Result<_, _>>()
}
pub fn puzzle_1(input: &str) -> Result<u32, PuzzleErr> {
Ok(parse_input(input)?.iter().map(|c| c.score()).sum())
}
fn count_cards(cards: &HashMap<u32, Card>, count: &mut HashMap<u32, u32>) {
for i in (1_u32..=cards.len() as u32).rev() {
let c = cards.get(&i).unwrap();
let new_ids: Vec<u32> = match c.n_hits() {
0 => Vec::new(),
x => Vec::from_iter((c.id + 1)..=(c.id + x)),
};
let n_new_cards: u32 =
new_ids.iter().map(|i| count.get(i).unwrap()).sum::<u32>() + new_ids.len() as u32;
count.insert(c.id, n_new_cards);
}
}
pub fn puzzle_2(input: &str) -> Result<u32, PuzzleErr> {
let cards = parse_input(input)?
.iter()
.map(|c| (c.id, c.clone()))
.collect::<HashMap<_, _>>();
let mut counts: HashMap<u32, u32> = HashMap::new();
count_cards(&cards, &mut counts);
Ok(counts.values().sum::<u32>() + cards.len() as u32)
}
pub fn main(data_dir: &str) {
println!("Day 4: Scratchcards");
let data = load(data_dir, 4, None);
// Puzzle 1.
let answer_1 = puzzle_1(&data);
match answer_1 {
Ok(x) => println!(" Puzzle 1: {}", x),
Err(e) => panic!("No solution to puzzle 1: {}.", e),
}
assert_eq!(answer_1, Ok(21158));
// Puzzle 2.
let answer_2 = puzzle_2(&data);
match answer_2 {
Ok(x) => println!(" Puzzle 2: {}", x),
Err(e) => panic!("No solution to puzzle 2: {}", e),
}
assert_eq!(answer_2, Ok(6050769))
}