This commit is contained in:
Moritz Hölting 2023-12-07 12:04:51 +01:00
parent 7354c7f628
commit b839a6f82d
6 changed files with 1328 additions and 1 deletions

8
Cargo.lock generated
View File

@ -86,6 +86,14 @@ dependencies = [
"rayon", "rayon",
] ]
[[package]]
name = "day-07"
version = "0.0.0"
dependencies = [
"indoc",
"itertools",
]
[[package]] [[package]]
name = "either" name = "either"
version = "1.9.0" version = "1.9.0"

View File

@ -5,7 +5,8 @@ members = [
"day-02", "day-02",
"day-03", "day-03",
"day-05", "day-05",
"day-06" "day-06",
"day-07"
] ]
[workspace.dependencies] [workspace.dependencies]

10
day-07/Cargo.toml Normal file
View File

@ -0,0 +1,10 @@
[package]
name = "day-07"
version = "0.0.0"
edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[dependencies]
indoc.workspace = true
itertools.workspace = true

1000
day-07/src/bin/input.txt Normal file

File diff suppressed because it is too large Load Diff

150
day-07/src/bin/part1.rs Normal file
View File

@ -0,0 +1,150 @@
use itertools::Itertools;
use std::cmp::Ordering;
fn main() {
println!("{}", part1(include_str!("./input.txt")));
}
fn part1(input: &str) -> u64 {
let hands = input.lines().map(process_hand).collect::<Vec<_>>();
calc_total_winnings(hands)
}
fn process_hand(input: &str) -> Hand {
let (hand, bid) = input.split_once(' ').expect("invalid input");
let cards = hand.chars().map(Card::from).collect::<Vec<_>>();
let mut card_amounts = cards.clone();
card_amounts.sort();
card_amounts.reverse();
let grouped_cards = card_amounts.into_iter().group_by(|c| *c);
let mut grouped_cards = grouped_cards
.into_iter()
.map(|(c, g)| (c, g.count()))
.collect::<Vec<_>>();
grouped_cards.sort_by(|(c1, g1), (c2, g2)| match g2.cmp(g1) {
Ordering::Equal => c2.cmp(c1),
x => x,
});
let mut grouped_cards = grouped_cards.into_iter();
let rank = match grouped_cards.next() {
Some((_, 5)) => Rank::FiveOfAKind,
Some((_, 4)) => Rank::FourOfAKind,
Some((_, 3)) => match grouped_cards.next() {
Some((_, 2)) => Rank::FullHouse,
Some((_, 1)) => Rank::ThreeOfAKind,
_ => panic!("invalid hand"),
},
Some((_, 2)) => match grouped_cards.next() {
Some((_, 2)) => Rank::TwoPair,
Some((_, 1)) => Rank::OnePair,
_ => panic!("invalid hand"),
},
Some((_, 1)) => Rank::HighCard,
Some((_, _)) => panic!("invalid hand"),
None => panic!("invalid hand"),
};
Hand {
bid: bid.parse().expect("invalid bid"),
cards: cards.try_into().expect("invalid hand"),
rank,
}
}
fn calc_total_winnings(mut hands: Vec<Hand>) -> u64 {
hands.sort();
hands
.into_iter()
.enumerate()
.map(|(i, h)| (i + 1) as u64 * h.bid as u64)
.sum()
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
enum Card {
N2,
N3,
N4,
N5,
N6,
N7,
N8,
N9,
T,
J,
Q,
K,
A,
}
impl From<char> for Card {
fn from(c: char) -> Self {
match c {
'2' => Self::N2,
'3' => Self::N3,
'4' => Self::N4,
'5' => Self::N5,
'6' => Self::N6,
'7' => Self::N7,
'8' => Self::N8,
'9' => Self::N9,
'T' => Self::T,
'J' => Self::J,
'Q' => Self::Q,
'K' => Self::K,
'A' => Self::A,
_ => panic!("invalid card"),
}
}
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
enum Rank {
HighCard,
OnePair,
TwoPair,
ThreeOfAKind,
FullHouse,
FourOfAKind,
FiveOfAKind,
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
struct Hand {
rank: Rank,
cards: [Card; 5],
bid: u32,
}
impl From<&str> for Hand {
fn from(s: &str) -> Self {
process_hand(s)
}
}
#[cfg(test)]
mod tests {
use super::*;
use indoc::indoc;
#[test]
fn test_part1() {
assert_eq!(
part1(indoc!(
"
32T3K 765
T55J5 684
KK677 28
KTJJT 220
QQQJA 483
"
)),
6440
);
}
}

158
day-07/src/bin/part2.rs Normal file
View File

@ -0,0 +1,158 @@
use itertools::Itertools;
use std::cmp::Ordering;
fn main() {
println!("{}", part2(include_str!("./input.txt")));
}
fn part2(input: &str) -> u64 {
let hands = input.lines().map(process_hand).collect::<Vec<_>>();
calc_total_winnings(hands)
}
fn process_hand(input: &str) -> Hand {
let (hand, bid) = input.split_once(' ').expect("invalid input");
let cards = hand.chars().map(Card::from).collect::<Vec<_>>();
let mut card_amounts = cards.clone();
card_amounts.sort();
card_amounts.reverse();
let grouped_cards = card_amounts.into_iter().group_by(|c| *c);
let mut grouped_cards = grouped_cards
.into_iter()
.filter(|(c, _)| *c != Card::J)
.map(|(c, g)| (c, g.count()))
.collect::<Vec<_>>();
grouped_cards.sort_by(|(c1, g1), (c2, g2)| match g2.cmp(g1) {
Ordering::Equal => c2.cmp(c1),
x => x,
});
let jokers = cards.iter().filter(|c| **c == Card::J).count();
let mut grouped_cards = grouped_cards.into_iter();
let rank = if let Some((_, a)) = grouped_cards.next() {
match a + jokers {
5 => Rank::FiveOfAKind,
4 => Rank::FourOfAKind,
3 => match grouped_cards.next() {
Some((_, 2)) => Rank::FullHouse,
Some((_, 1)) => Rank::ThreeOfAKind,
_ => panic!("invalid hand"),
},
2 => match grouped_cards.next() {
Some((_, 2)) => Rank::TwoPair,
Some((_, 1)) => Rank::OnePair,
_ => panic!("invalid hand"),
},
1 => Rank::HighCard,
_ => panic!("invalid hand"),
}
} else if jokers == 5 {
Rank::FiveOfAKind
} else {
panic!("invalid hand")
};
Hand {
bid: bid.parse().expect("invalid bid"),
cards: cards.try_into().expect("invalid hand"),
rank,
}
}
fn calc_total_winnings(mut hands: Vec<Hand>) -> u64 {
hands.sort();
hands
.into_iter()
.enumerate()
.map(|(i, h)| (i + 1) as u64 * h.bid as u64)
.sum()
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
enum Card {
J,
N2,
N3,
N4,
N5,
N6,
N7,
N8,
N9,
T,
Q,
K,
A,
}
impl From<char> for Card {
fn from(c: char) -> Self {
match c {
'2' => Self::N2,
'3' => Self::N3,
'4' => Self::N4,
'5' => Self::N5,
'6' => Self::N6,
'7' => Self::N7,
'8' => Self::N8,
'9' => Self::N9,
'T' => Self::T,
'J' => Self::J,
'Q' => Self::Q,
'K' => Self::K,
'A' => Self::A,
_ => panic!("invalid card"),
}
}
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
enum Rank {
HighCard,
OnePair,
TwoPair,
ThreeOfAKind,
FullHouse,
FourOfAKind,
FiveOfAKind,
}
#[derive(Debug, PartialEq, Clone, Copy, Eq, PartialOrd, Ord)]
struct Hand {
rank: Rank,
cards: [Card; 5],
bid: u32,
}
impl From<&str> for Hand {
fn from(s: &str) -> Self {
process_hand(s)
}
}
#[cfg(test)]
mod tests {
use super::*;
use indoc::indoc;
#[test]
fn test_part2() {
assert_eq!(
part2(indoc!(
"
32T3K 765
T55J5 684
KK677 28
KTJJT 220
QQQJA 483
"
)),
5905
);
}
}