当前位置: 首页
编程语言
C++ random_shuffle随机洗牌 _ 数组乱序打乱算法【实战】

C++ random_shuffle随机洗牌 _ 数组乱序打乱算法【实战】

热心网友 时间:2026-05-06
转载

C++17 中 std::random_shuffle 已被移除:全面转向 std::shuffle 与确定性随机引擎指南

C++ random_shuffle随机洗牌 _ 数组乱序打乱算法【实战】

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

C++17 标准已移除 random_shuffle:为何必须停止使用

一个明确的结论是:std::random_shuffle 函数已在 C++17 标准中被彻底删除。如果你在使用 GCC 9+ 或 Clang 7+ 等现代编译器,并启用了 -std=c++17 或更高标准,将会直接遇到编译错误:error: 'random_shuffle' is not a member of 'std'。事实上,该函数在 C++11 中已被标记为“废弃”(deprecated)。主要原因在于其内部依赖全局的 std::rand 函数,这导致随机状态不可预测、结果无法重现、多线程环境下不安全,并且最关键的是,开发者无法为其指定高质量的随机数生成引擎。

如何正确使用 std::shuffle 与 std::mt19937 进行替代

标准的替代方案是使用 std::shuffle 并搭配一个高质量的随机数引擎,例如最常用的 std::mt19937。这里的关键点不仅仅是更换函数名,核心在于你必须显式地传入一个可调用的随机数生成器对象

  • std::shuffle 的第三个参数必须是一个函数对象(functor)或 lambda 表达式,直接使用函数指针或 std::rand 是无效的。
  • 初始化 std::mt19937 需要一个种子(seed)。通常建议使用 std::random_device 来生成这个种子,以确保每次程序运行能产生不同的随机序列。
  • 该方法通用性强,适用于 std::array、原生 C 风格数组以及 std::vector 等多种容器,但务必注意传递正确的迭代器范围(beginend)。

以下是一个打乱整型数组的典型代码示例:

#include 
#include 
#include 

std::array arr = {1, 2, 3, 4, 5};
std::random_device rd;
std::mt19937 g(rd());  // 注意:g 是生成器对象,不是类型
std::shuffle(arr.begin(), arr.end(), g);

原生 C 风格数组如何安全地进行 shuffle 操作?

对于 int arr[10] 这类原生数组,一个常见的错误是直接写成 std::shuffle(arr, arr+10, g)。虽然语法上可能通过,但存在潜在风险:如果该数组作为函数参数传递(此时会退化为指针),那么 sizeof(arr) 将不再是数组长度 10,计算出的 arr+10 很可能导致越界访问。

更安全的做法有两种:一是利用 C++20 的 std::span;二是使用 std::beginstd::end 这类非成员函数:

  • 在 C++11/14/17 中,可以写成 std::shuffle(std::begin(arr), std::end(arr), g),这依赖于 ADL(参数依赖查找)来正确推导数组长度。
  • 如果需要兼容旧标准或进行手动控制,可以先定义 constexpr size_t N = 10;,然后使用 arr + N 作为结束迭代器。
  • 这里有一个必须避免的陷阱:绝对不要在函数内部对形式参数数组使用 sizeof 来计算其长度

如何复现特定的打乱结果?固定种子即可实现

在调试或进行单元测试时,经常需要复现某一次特定的乱序结果。实现方法很简单:只需固定 std::mt19937 的种子即可,例如 std::mt19937 g(42)。这种方式比传统的 srand(42); random_shuffle(...) 组合更加可靠,因为 std::shuffle 采用的 Fisher–Yates 算法本身是确定性的,且随机数引擎的行为完全由种子决定。

但需要注意两个细节:首先,同一个 g 对象如果用于多次 shuffle 调用,那么只有第一次调用是基于初始种子的确定序列,后续调用会基于引擎剩余的内部状态进行,结果虽可重现但可能不符合“每次重新开始”的预期。因此,如果需要反复打乱同一个容器并希望每次都是独立且可重现的序列,要么每次都重新构造一个 g 对象,要么调用 g.seed(new_seed) 来重置其状态。

最后,还有一个容易被忽略的细节:std::shuffle 本身被设计为不抛出异常,但如果传入无效的迭代器范围(例如 begin > end),其行为是未定义的。此外,在某些嵌入式或受限环境中,std::random_device 可能因无法访问真随机源而抛出 std::runtime_error。对于生产环境,建议加入 try-catch 块进行处理,或准备一个基于时间戳的备用种子生成方案。

来源:https://www.php.cn/faq/2313807.html

游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

同类文章
更多
Go语言Gin怎么做参数校验_Go语言Gin Validator校验教程【秒懂】

Go语言Gin怎么做参数校验_Go语言Gin Validator校验教程【秒懂】

Gin框架binding: "required "校验失效的常见原因与解决方案:绑定方式、Content-Type匹配及嵌套结构处理详解 为什么Gin框架中binding: "required "标签有时会失效? 在Go语言的Gin框架开发中,参数校验是保障接口健壮性的关键环节。许多开发者初次使用bindi

时间:2026-05-06 07:48
c++如何实现文件追加写入_ios::app标志位使用详解【代码】

c++如何实现文件追加写入_ios::app标志位使用详解【代码】

std::ios::app 是最可靠的追加写入方式,强制所有写入发生在文件末尾且不受 seekp() 影响;仅用 std::ios::out 会清空文件,std::ios::ate 则不保证追加语义。 用 std::ofstream 打开文件时加 std::ios::app 就能追加写入 核心结论:

时间:2026-05-06 07:47
如何在PHP中从文本文件随机读取带变量的模板行

如何在PHP中从文本文件随机读取带变量的模板行

PHP实现文本模板随机读取与变量动态替换的完整指南 本文详解一种高效安全的PHP模板处理方案:通过预设占位符(如{TITLE})构建纯文本模板,结合str_replace()函数实现变量动态注入,彻底规避直接执行PHP代码可能引发的安全漏洞与语法解析错误。 在PHP网站开发与内容管理实践中,开发者经

时间:2026-05-06 07:47
C++判断字符串是否全为英文字母 _ isalpha函数循环检查【实战】

C++判断字符串是否全为英文字母 _ isalpha函数循环检查【实战】

C++判断字符串是否全为英文字母:避开 isalpha 函数的常见陷阱与最佳实践 在C++编程中,判断一个字符串是否完全由英文字母组成,看似是一个基础任务。许多开发者会下意识地想到使用循环配合 std::isalpha 函数逐个检查字符。然而,这种直接的方法极易引发未定义行为、编码误解和边界条件处理

时间:2026-05-06 07:47
FastAPI 密码校验错误未按预期返回自定义 HTTP 错误的解决方案

FastAPI 密码校验错误未按预期返回自定义 HTTP 错误的解决方案

FastAPI 密码校验错误未按预期返回自定义 HTTP 错误的解决方案 在 FastAPI 开发中,使用 Pydantic v2 的 constr(min_length=6) 等字段约束会触发自动的 422 响应,导致自定义的 HTTPException 无法生效。正确的解决方案是移除字段级的约束

时间:2026-05-06 07:47
热门专题
更多
刀塔传奇破解版无限钻石下载大全 刀塔传奇破解版无限钻石下载大全
洛克王国正式正版手游下载安装大全 洛克王国正式正版手游下载安装大全
思美人手游下载专区 思美人手游下载专区
好玩的阿拉德之怒游戏下载合集 好玩的阿拉德之怒游戏下载合集
不思议迷宫手游下载合集 不思议迷宫手游下载合集
百宝袋汉化组游戏最新合集 百宝袋汉化组游戏最新合集
jsk游戏合集30款游戏大全 jsk游戏合集30款游戏大全
宾果消消消原版下载大全 宾果消消消原版下载大全
  • 日榜
  • 周榜
  • 月榜
热门教程
更多
  • 游戏攻略
  • 安卓教程
  • 苹果教程
  • 电脑教程