登录
首页 >  文章 >  java教程

标准扑克牌洗牌与排序技巧

时间:2026-01-09 21:33:43 170浏览 收藏

在文章实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天golang学习网就整理分享《标准扑克牌如何正确洗牌与排序方法》,聊聊,希望可以帮助到正在努力赚钱的你。

如何正确实现并排序一副标准扑克牌(52张)

本文详解如何构建完整的 `Deck` 类,初始化52张不重复的标准扑克牌,并通过 `Comparable` 接口和 `Arrays.sort()` 实现按点数升序排列,解决手动插入混乱、数组未填满、排序逻辑错误等常见问题。

要构建一个真正可用的扑克牌系统,关键在于两点:正确生成完整、无重复的52张牌,以及支持可预测、可复现的有序排列。当前代码存在多个根本性问题——Deck 构造器仅创建了5张随机牌(而非52张),且未保证唯一性;Card 类的构造器使用 Random 随机生成,导致无法构建标准有序牌堆;自定义插入排序逻辑混入了 Collections.shuffle(),造成“先排后洗”的逻辑矛盾;而 Card.SortCards() 中对 RANKS 数组调用 Arrays.sort() 更是无效操作(RANKS 是 String[] 常量,且字典序排序 "10" 会排在 "2" 前)。

✅ 正确实现方案

1. 改进 Card 类:实现 Comparable 并固定构造逻辑

public class Card implements Comparable<Card> {
    private final int rank; // 0=2, 1=3, ..., 12=Ace
    private final int suit; // 0=Clubs, 1=Diamonds, 2=Hearts, 3=Spades

    private static final String[] RANKS = {"2", "3", "4", "5", "6", "7", "8", "9", "10", "Jack", "Queen", "King", "Ace"};
    private static final String[] SUITS = {"Clubs", "Diamonds", "Hearts", "Spades"};

    // 明确指定花色与点数的构造器(用于构建标准牌堆)
    public Card(int rank, int suit) {
        if (rank < 0 || rank >= RANKS.length) throw new IllegalArgumentException("Invalid rank: " + rank);
        if (suit < 0 || suit >= SUITS.length) throw new IllegalArgumentException("Invalid suit: " + suit);
        this.rank = rank;
        this.suit = suit;
    }

    // 保留随机构造器(仅用于发牌/测试,不用于初始化整副牌)
    public Card() {
        this((int)(Math.random() * RANKS.length), (int)(Math.random() * SUITS.length));
    }

    public String getRank() { return RANKS[rank]; }
    public String getSuit() { return SUITS[suit]; }
    public int getRankValue() { return rank; }

    @Override
    public int compareTo(Card o) {
        return Integer.compare(this.rank, o.rank); // 按点数升序(2→Ace)
    }

    @Override
    public String toString() {
        return getRank() + " of " + getSuit();
    }
}

? 关键改进:Card 不再依赖内部 Random 初始化整副牌;compareTo() 确保自然排序逻辑清晰可靠。

2. 重构 Deck 类:完整初始化 + 安全排序

import java.util.Arrays;

public class Deck {
    private final Card[] cards;

    public Deck() {
        this.cards = new Card[52];
        int index = 0;
        for (int suit = 0; suit < 4; suit++) {
            for (int rank = 0; rank < 13; rank++) {
                cards[index++] = new Card(rank, suit); // 严格按顺序生成:2♣, 3♣, ..., A♠
            }
        }
    }

    // 标准升序排序(按点数,同点数不区分花色)
    public void sort() {
        Arrays.sort(cards);
    }

    // 安全排序(兼容含 null 的稀疏数组,如只取前 n 张时)
    public void sortSafely() {
        Arrays.sort(cards, (c1, c2) -> {
            if (c1 == null && c2 == null) return 0;
            if (c1 == null) return 1;
            if (c2 == null) return -1;
            return c1.compareTo(c2);
        });
    }

    // 发牌:返回前 n 张(n ∈ [5,10]),自动截断并返回新数组
    public Card[] deal(int n) {
        if (n < 5 || n > 10) throw new IllegalArgumentException("Deal count must be between 5 and 10");
        Card[] hand = new Card[n];
        System.arraycopy(cards, 0, hand, 0, n);
        return hand;
    }

    @Override
    public String toString() {
        return Arrays.toString(cards);
    }
}

✅ Deck() 构造器现在确定性生成52张唯一牌,顺序为:所有黑桃→红桃→方块→梅花,每种花色内按 2→A 升序。这是后续排序和调试的基础。

3. 使用示例:生成、排序、发牌、打印

public class CardGame {
    public static void main(String[] args) {
        Deck deck = new Deck();
        System.out.println("【初始牌堆(有序)】");
        System.out.println(Arrays.toString(deck.deal(5))); // [2 of Clubs, 3 of Clubs, 4 of Clubs, 5 of Clubs, 6 of Clubs]

        // 洗牌(可选)
        shuffle(deck.cards);

        System.out.println("\n【洗牌后】");
        System.out.println(Arrays.toString(deck.deal(5)));

        // 再次排序(恢复升序)
        deck.sort();
        System.out.println("\n【重新排序后】");
        System.out.println(Arrays.toString(deck.deal(5)));
    }

    // 简易 Fisher-Yates 洗牌(替代 Collections.shuffle,避免 ArrayList 转换开销)
    private static void shuffle(Card[] arr) {
        for (int i = arr.length - 1; i > 0; i--) {
            int j = (int)(Math.random() * (i + 1));
            Card temp = arr[i];
            arr[i] = arr[j];
            arr[j] = temp;
        }
    }
}

⚠️ 注意事项与最佳实践

  • 不要在 Deck 中混用随机生成与有序构建:整副牌应由确定性逻辑生成,随机性仅应用于洗牌或抽牌阶段;
  • 避免 null 元素陷阱:若需动态管理手牌(如出牌后移除),建议改用 ArrayList 替代固定数组;
  • 扩展排序逻辑:如需按「点数+花色」双重排序(如 2♣ < 2♦ < 2♥ < 2♠),可增强 compareTo():
    @Override
    public int compareTo(Card o) {
        int rankCmp = Integer.compare(this.rank, o.rank);
        if (rankCmp != 0) return rankCmp;
        return Integer.compare(this.suit, o.suit); // 同点数时按花色排序
    }
  • 性能提示:Arrays.sort() 对对象数组使用的是双轴快排(Timsort 变种),时间复杂度 O(n log n),远优于手写插入排序,且更稳定可靠。

通过以上重构,你将获得一副可预测、可排序、可扩展的标准扑克牌系统,为后续实现发牌、比大小、玩家手牌管理等游戏逻辑打下坚实基础。

今天关于《标准扑克牌洗牌与排序技巧》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>