Java双色球摇号器实现:Set去重与随机数应用
时间:2026-05-14 14:14:42 237浏览 收藏
本文深入剖析了Java中双色球摇号器的核心实现逻辑,聚焦于如何用HashSet正确实现红球去重与随机性保障(严防TreeSet自动排序破坏随机本质),精准规避Random.nextInt()边界陷阱(如红球需nextInt(33)+1而非nextInt(33)),并厘清红蓝球池严格分离、循环终止条件必须基于size()==6等关键细节;同时强调生成后排序与格式化(补零、升序展示)仅用于输出,绝不干扰随机过程本身——真正考验程序员的,不是写几行随机代码,而是对业务规则(范围、无序、去重、显示规范)毫厘不差的理解与落地。

为什么用 Set 存红球但不能直接用 TreeSet
因为双色球红球要求“不重复、无序、范围 1–33”,Set 确实能去重,但选错实现类会出问题:TreeSet 自动排序,生成的号码永远是 1,2,3,4,5… 这不符合“随机摇出”的业务逻辑。必须用 HashSet 或 LinkedHashSet(后者保留插入顺序,但摇号不需要),否则你写的“随机”只是假象。
实操建议:
- 声明时用
Set,别写redBalls = new HashSet(); new TreeSet() - 插入前不用手动判断是否已存在——
Set.add()返回false表示重复,可用来控制循环退出条件 - 别在
while循环里反复new HashSet(),那会导致每次都是空集合,永远加不满 6 个
Random.nextInt() 的边界陷阱:为什么总摇不出 33 号红球
Random.nextInt(33) 返回的是 0–32,不是 1–33。这是新手最常踩的坑——直接套公式却忽略偏移量。蓝球同理,nextInt(16) 是 0–15,而规则要求 1–16。
正确做法:
- 红球:
random.nextInt(33) + 1 - 蓝球:
random.nextInt(16) + 1 - 别用
Math.random() * 33再强转int,类型转换截断行为不稳定,且可读性差 - 如果用
ThreadLocalRandom(推荐多线程场景),调用方式是ThreadLocalRandom.current().nextInt(1, 34),注意右边界是开区间
如何保证“6 个红球 + 1 个蓝球”稳定输出,而不是有时 5 个有时 7 个
核心在于控制循环终止条件。很多人写 while (set.size() 却忘了红球和蓝球要分开生成;更隐蔽的问题是:用同一个 Random 实例连续生成时,没考虑蓝球是否可能撞上某个红球(虽然规则允许,但逻辑上应完全独立)。
关键点:
- 红球用一个
Set,蓝球用单独一个int变量,二者绝不混用容器 - 红球循环用
while (redBalls.size() != 6)比更直白,避免脑内换算 - 不要把蓝球也塞进红球的
Set里去重——规则明确红蓝球池分离 - 生成完立即转成
List并用Collections.sort()排红球(显示需要升序),但排序必须在生成完毕后做,不能边插边排
从 Set 到最终输出:为什么 toString() 直接打印看起来像乱序但其实是正常的
HashSet 不保证迭代顺序,所以 System.out.println(redBalls) 输出可能是 [12, 3, 27, 8, 1, 19]。这不是 bug,是预期行为。但用户看到“摇号结果”期望是整齐的、带格式的,比如“红球:01 03 08 12 19 27|蓝球:09”。
处理建议:
- 红球转
List后排序:ListsortedRed = new ArrayList(redBalls); Collections.sort(sortedRed); - 格式化输出时补零:
String.format("%02d", num),不然 1 号球显示成 “1” 而不是 “01” - 别对
Set直接调用stream().sorted()后 collect 回Set——又绕回去了,失去排序意义
真正难的不是生成随机数,而是让每一次“摇”都符合规则细节:范围、去重粒度、红蓝隔离、显示规范。少盯代码行数,多看双色球开奖公告的格式和数字范围——那才是唯一正确的测试用例。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Java双色球摇号器实现:Set去重与随机数应用》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
248 收藏
-
293 收藏
-
237 收藏
-
328 收藏
-
268 收藏
-
312 收藏
-
315 收藏
-
362 收藏
-
141 收藏
-
239 收藏
-
347 收藏
-
218 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习