Advent of Code 2023 — Day 7
Day 7 saw us playing cards, on camels — camel cards!
Camel cards are like poker. But simpler, so you can play it while on a camel.
The hand-rules are similar to Poker, but with a few changes — which is ace, because I have no idea what the poker rules are anymore.
Five of a kind, where all five cards have the same label:
AAAAA
Four of a kind, where four cards have the same label and one card has a different label:
AA8AA
Full house, where three cards have the same label, and the remaining two cards share a different label:
23332
Three of a kind, where three cards have the same label, and the remaining two cards are each different from any other card in the hand:
TTT98
Two pair, where two cards share one label, two other cards share a second label, and the remaining card has a third label:
23432
One pair, where two cards share one label, and the other three cards have a different label from the pair and each other:
A23A4
High card, where all cards' labels are distinct:
23456
The hands are also ranked by suit, in descending order where an ace is high.
Part 1
In part 1, we have the following input:
32T3K 765
T55J5 684
KK677 28
KTJJT 220
QQQJA 483
Where the right hand column is the bid amount for the hand.
Our job is to sort each hand, and then multiply each hand's bid with its rank.
So if we take the order above as an example — we'd have 765 * 1
+ 684 * 2
+ ...
I started off this puzzle by assigning a suit / card number a letter (easier to sort alphabetically than numerically in this case):
ORDER = {"A" => "a", "K" => "b", "Q" => "c", "J" => "d", "T" => "e", "9" => "f", "8" => "g", "7" => "h", "6" => "i", "5" => "j", "4" => "k", "3" => "l", "2" => "m"}.freeze
Then turned the input in to an array of arrays that looked like [[hand, bid], ...]
, sorted the list and then went to work adding up each bid:
rounds = input.map { |l| l.split(" ") }
sorted_rounds = rounds
.sort {|a,b| [hand_type(a[0]), hand_value(b[0])] <=> [hand_type(b[0]), hand_value(a[0])] }
sorted_rounds
.map
.with_index(1) {|r, idx| r[1].to_i * idx }
.sum
To find the type of hand I was looking at, I used the below 2 methods.
First, I turned the hand in to a hash with the number of cards of each type. So for an input of K3558
I would have {K => 1, 3 => 1, 5 => 2, 8 => 1}
. Then, I take the values and sort them — in to something like [2,1,1,1]
.
This is then passed in to the score_hand
method which returns a rank based on the hand.
def hand_type(hand)
cards = hand.gsub(/(.)\0*/).to_a
sorted_cards = cards.tally.values.sort {|a,b| b <=> a}
score_hand(sorted_cards)
end
def score_hand(cards)
return 6 if cards == [5]
return 5 if cards == [4,1]
return 4 if cards == [3,2]
return 3 if cards == [3,1,1]
return 2 if cards == [2,2,1]
return 1 if cards == [2,1,1,1]
return 0 if cards == [1,1,1,1,1]
end
Lastly, we need to sort cards that have the same type based on the order of the cards in hand.
For this, sorting alphabetically was was simpler than zipping each array and comparing card values.
For this, I took a look at the ORDER
constant and turned a hand like K3558
to bljjg
, so when compared with another hand that has the same type, say K3448
(bliig
), the first hand wins.
def hand_value(hand)
hand.gsub(/(.)\0*/).map{ |v| ORDER[v] }.join("")
end
Part 2
In part 2, Jacks are out, Jokers are in.
Jokers are the lowest scoring card, but they can be used as a wildcard to improve a hand to one of a better quality. So for example, the hand 33JJ8
goes from being two of a kind [33, JJ]
to four of a kind [3333]
!
Because Jokers are now worthless, I have a new order for them:
IMPROVED_ORDER = {"A" => "a", "K" => "b", "Q" => "c", "T" => "d", "9" => "e", "8" => "f", "7" => "g", "6" => "h", "5" => "i", "4" => "j", "3" => "k", "2" => "l", "J" => "m"}.freeze
The code for part 2 and part 1 is pretty much the same, except I try and improve each hand when sorting:
rounds = input
.map { |l| l.split(" ") }
sorted_rounds = rounds
.sort {|a,b| [improve_hand(a[0]), improved_hand_value(b[0])] <=> [improve_hand(b[0]), improved_hand_value(a[0])] }
sorted_rounds
.map
.with_index(1) {|r, idx| r[1].to_i * idx }
.sum
I try and improve each hand with the method below. I do the same tallying of each hand.
So 33JJ8
becomes {3 => 2, J => 2, 8 => 1}
.
But then I take any J
cards and add them to whichever card type has the most already.
So now, the tally looks like {3 => 4, 8 => 1}
— which makes this a 4 of a kind.
Thankfully Jokers only improve hand quality, not overall score, so we didn't need to consider hands like KQKQJ
and QKQKJ
and apply the Joker to the King in order to get a better overall score.
def improve_hand(hand)
cards = hand.gsub(/(.)\0*/).to_a
return hand_type(hand) unless cards.any? { |c| c == "J" }
return hand_type(hand) if cards.all? { |c| c == "J" }
card_types = cards.tally
improve_by = card_types["J"]
type_to_improve = type_to_improve(card_types)
card_types[type_to_improve] += improve_by if improve_by && type_to_improve
card_types.delete("J")
sorted_cards = card_types.values.sort {|a,b| b <=> a}
score_hand(sorted_cards)
end
def type_to_improve(cards)
return nil if cards.empty?
no_jack = cards.reject{|k,v| k == "J" }
return nil if no_jack.empty?
no_jack.max_by{|_,v| v}[0]
end
⭐️ Success!
Day 7: https://github.com/dNitza/advent-of-code/blob/main/2023/07/lib/puzzle.rb
Performance
day 07
├─ part 1 — 0.073877s
╰─ part 2 — 0.101986s
Tags: advent of code ruby