正确生成不重复随机数需用洗牌算法或ThreadLocalRandom;避免重复种子,复用Random实例,多线程优选ThreadLocalRandom,唯一性需求结合UUID或时间戳。
Java中生成不同的随机数,关键在于正确使用随机数生成器,并避免重复种子或不当调用。最常用、推荐的方式是使用 java.util.Random 或更现代的 ThreadLocalRandom(多线程场景下更优),而不是反复创建新实例或用 Math.random() 做简单拼凑。
如果目标是在一个固定范围内(比如 1~100)获取若干个互不相同的随机数,不能靠“多次 nextInt() 就一定不重复”——那是概率问题,不是保证。正确做法是:
Collections.shuffle(list) 打乱顺序示例:获取 5 个 1~20 之间不重复的随机整数
Listnumbers = new ArrayList<>(); for (int i = 1; i <= 20; i++) numbers.add(i); Collections.shuffle(numbers); List result = numbers.subList(0, 5); // [17, 3, 19, 8, 12]
在并发环境中,Random 是线程安全但有竞争开销;ThreadLocalRandom 为每个线程提供独立实例,性能更好,且用法简洁:
ThreadLocalRandom.current().nextInt(1, 101) → 1~100 的随机 int(右边界不包含)ThreadLocalRandom.
current().nextLong(1000, 9999) → 四位随机长整型ThreadLocalRandom.current().nextDouble(0.0, 1.0) → [0.0, 1.0) 区间 double注意:它不保证多次调用结果不重复,只是每次调用都独立、统计上均匀——这正是“随机”的本意。若需“不重复”,仍要配合洗牌或 Set 去重逻辑。
这是新手易犯的坑:在循环里反复用当前时间戳做种子创建 Random 实例:
// ❌ 错误示范:高概率生成相同数字
for (int i = 0; i < 5; i++) {
Random r = new Random(System.currentTimeMillis()); // 毫秒级,循环太快,时间戳一样!
System.out.println(r.nextInt(100));
}
正确做法是:只创建一次 Random 实例(复用),或直接用静态工具类如 ThreadLocalRandom。
如果“不同”指的是全局唯一(比如生成订单号、ID),单纯随机数不够可靠。这时应优先考虑:
UUID.randomUUID() → 生成 128 位唯一标识符,碰撞概率极低System.nanoTime() + "-" + ThreadLocalRandom.current().nextInt(10000)
纯随机 ≠ 唯一,唯一需设计保障,随机只是辅助手段。
基本上就这些。核心就两点:范围去重要洗牌,高频/并发用 ThreadLocalRandom,别手抖 new 太多 Random。