0

0

C++ 怎么生成不重复随机数 C++ shuffle算法打乱数组顺序【逻辑】

尼克

尼克

发布时间:2026-01-29 11:01:02

|

431人浏览过

|

来源于php中文网

原创

最可靠方案是用 std::shuffle 配合 std::mt19937 引擎打乱已填充的容器,本质为 Fisher-Yates 算法,天然不重复且均匀;禁用 rand() 和手写查重循环,容器须支持随机访问迭代器。

c++ 怎么生成不重复随机数 c++ shuffle算法打乱数组顺序【逻辑】

std::shuffle 生成不重复随机数最可靠

直接结论:不要手写“while 循环 + set 查重”,也不要用 rand() 配模运算——既低效又易偏态。C++11 起,std::shuffle 是标准、高效、无偏的首选方案,本质是 Fisher-Yates 洗牌算法,天然保证不重复。

它不生成“随机数”,而是打乱一个已知不重复的序列(比如 0,1,2,...,n-1),所以结果必然不重复,且均匀分布。

  • 必须传入真随机数引擎,如 std::mt19937,不能用默认构造的 std::default_random_engine(种子固定,每次运行结果一样)
  • 容器必须支持随机访问迭代器,std::vectorstd::array 可用,std::list 不行
  • 打乱前需先填充目标范围,例如要 10 个不重复的 1~100 之间的数,就先构造含 1~100 的 vector,再 shuffle 后取前 10 个

std::shuffle 的典型用法和常见错误

正确写法依赖三个要素:容器、随机引擎、适配器。漏掉任意一个都会出问题。

std::vector nums;
for (int i = 1; i <= 100; ++i) nums.push_back(i);
std::random_device rd;
std::mt19937 g(rd()); // 必须用 rd() 初始化,否则全是同一序列
std::shuffle(nums.begin(), nums.end(), g); // 注意:是 [begin, end),不是 end-1
  • 错误:用 std::rand 当第三个参数 → 编译失败,std::shuffle 要求引擎满足 UniformRandomBitGenerator 概念
  • 错误:写成 std::shuffle(v.begin(), v.end()-1, g) → 最后一个元素永远不动
  • 错误:在循环里反复创建 std::mt19937 并用相同种子 → 所有 shuffle 结果一模一样

如果只要几个不重复随机数,别 shuffle 整个大数组

比如从 1~10000 中抽 5 个不重复数,没必要构造一万个 int 再 shuffle。这时改用 std::sample(C++17)更省空间和时间:

知了zKnown
知了zKnown

知了zKnown:致力于信息降噪 / 阅读提效的个人知识助手。

下载

立即学习C++免费学习笔记(深入)”;

std::vector pool(10000);
std::iota(pool.begin(), pool.end(), 1); // 填 1~10000
std::vector result(5);
std::sample(pool.begin(), pool.end(), result.begin(), 5, 
            std::mt19937{std::random_device{}()});
  • std::sample 内部使用 reservoir sampling 或类似策略,时间复杂度 O(n),空间只占输出大小
  • 若编译器不支持 C++17,退回到先建小 pool(如 1~100)再 shuffle,或手动实现带 map 查重的抽取(但仅当 n 极小才划算)
  • 注意 std::sample 不保证结果有序,如需升序得额外调 std::sort

为什么不用 std::rand() % N 加 set 去重?

看似简单,实际在中等规模(比如抽 50 个不重复数,范围 0~99)时,碰撞概率陡增,性能崩塌。数学上期望尝试次数是调和级数,抽 k 个数在 N 范围内平均要调用 rand 约 N·H_k ≈ N·(ln k + γ) 次;k 接近 N 时,后期几乎每次都在撞。

  • 更严重的是 std::rand() 本身周期短、低位比特相关性高,% 运算会放大偏差,导致某些数永远比别的数更容易被选中
  • set 插入 + 查重带来 O(k log k) 额外开销,而 std::shuffle 是 O(k) 时间(只 shuffle 前 k 位即可)+ O(k) 空间
  • 没有线程安全性:std::rand() 是全局状态,多线程并发调用未定义行为;std::mt19937 实例是局部对象,天然安全

真正需要警惕的,是把“能跑通”当成“正确”——尤其在测试样本小的时候,偏差根本看不出来。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

391

2023.09.04

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

94

2023.09.25

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

463

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

544

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

93

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

503

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

166

2025.12.24

Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

0

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 7.8万人学习

C 教程
C 教程

共75课时 | 4.3万人学习

C++教程
C++教程

共115课时 | 14.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号