Project Euler #54: Poker hands

Sort by

recency

|

24 Discussions

|

  • + 0 comments

    See full explaination with commented code at Beta Projects Project Euler 54

    from collections import Counter
    values = {r: i for i, r in enumerate('23456789TJQKA', start=2)}
    straights = [(v, v-1, v-2, v-3, v-4) for v in range(14, 5, -1)] + [(5, 4, 3, 2, 1)]
    ranks = [(1, 1, 1, 1, 1),(2, 1, 1, 1),(2, 2, 1),(3, 1, 1),(),(),(3, 2),(4, 1)]
    def hand_rank(hand):
        counts = Counter(card[0] for card in hand)
        sorted_counts = sorted(((v, values[k]) for k, v in counts.items()), reverse=True)
        score = list(zip(*sorted_counts))
        try:
            score[0] = ranks.index(score[0])
        except ValueError:
            score[0] = 0  # Default to High card if not found
        if len(set(card[1] for card in hand)) == 1: score[0] = 5  # Flush rank
        current_straight = tuple(score[1])
        if current_straight == (14, 5, 4, 3, 2): 
            score[1] = (5, 4, 3, 2, 1)
            current_straight = (5, 4, 3, 2, 1)
        if current_straight in straights:
            score[0] = 8 if score[0] == 5 else 4  # 8: Straight Flush, 4: Straight
        return score
    for _ in range(int(input())):
        hand = input().split()
        player1, player2 = hand[:5], hand[5:]
        print("Player 1" if hand_rank(player1) > hand_rank(player2) else "Player 2")
    
  • + 0 comments

    Some points-
    1) You have to consider 10,Q,J,K,A too as consecutive cards as mentioned in last point in problem. And in it A is the highest card.
    2) You have to consider A,2,3,4,5 too as consecutive cards and in it use 5 as highest. (test case 6, 2)

  • + 0 comments

    100/- points python 3

    import collections
    
    numbers=['2','3','4','5','6','7','8','9','T','J','Q','K','A']
    #numbers_2=['A','2','3','4','5','6','7','8','9','T','J','Q','K']
    
    def ranker(cards):
        suits=[i[-1] for i in cards]
        if len(set(suits))==1:
            same_suit=True
        else:
            same_suit=False
        
        values=[numbers.index(i[0]) for i in cards]
        # values_2=[numbers_2.index(i[0]) for i in cards]
        values.sort(reverse=True)
        if values==[12,3,2,1,0]:
            if same_suit:
                return (8,3)
            else:
                return (4,3)
        # values_2.sort(reverse=True)
        differences=[values[i]-values[i+1] for i in range(4)]
        groups=[] # I could have grouped by using itertools.groupby but I wanted to write it alorithmically
        differences.append(-1) #pre-processing to obtain last interval too
        start=0
        for i in range(1,len(differences)):
            if differences[i]!=differences[i-1]:
                groups.append((differences[i-1],i-start))
                start=i
        
        if (1,4) in groups :
            consecutive_values=True
        else:
            consecutive_values=False
        
        values_same=[i[0] for i in collections.Counter(values).most_common()] #this is coool
    
        if same_suit and consecutive_values:
            return (8,max(values))
        elif (0,3) in groups:
            # if values[0]!=values[1]:
            #     if values[0]!=values[2]:
            #         ans=(values[1],values[0])
            #     else:
            #         ans=(values[0],values[1])
            # else:
            #     ans=(values[0],int(set(values).difference([values[0]])))
            # return (7,*ans)
            return (7,*values_same)        
        elif (0,1) in groups and (0,2) in groups: 
            return (6,*values_same)
        elif same_suit:
            return (5,*values)
        elif consecutive_values:
            return (4,max(values))
        elif (0,2) in groups:
            return (3,*values_same )
        elif groups.count((0,1))==2:
            return (2,*values_same)
        elif (0,1) in groups:
            return (1,*values_same)
        else:
            return (0,*values)
                    
    def sorter(pl1,pl2): 
        size=min(len(pl1),len(pl2))
        for i in range(size):
            if pl1[i]>pl2[i]:
                return 'Player 1'
            elif pl1[i]<pl2[i]:
                return 'Player 2'
            
    for _ in range(int(input())):
        cards=input().split()
        pl1=cards[:5]
        pl2=cards[5:]
        print(sorter(ranker(pl1),ranker(pl2)))
    
  • + 0 comments

    My code is right but still test cases is failed

  • + 1 comment

    It is usually the case that Aces can be high or low card depending on whether that is useful for making a straight. In which case a hand with A 2 3 4 5 is considered a 5 high straight. This is fairly standard for poker rules. It was not mentioned in the problem statement, but using this option on the Ace is necessary to pass all of the test cases, specifically case 6.