rand()总生成相同随机数因未设种子或秒级时间种子重复;C++11后应使用<random>库,配合std::random_device、std::mt19937和std::uniform_int_distribution实现真随机、均匀分布、线程安全及跨平台可靠随机数。

为什么 rand() 生成的随机数总是一样的?
因为没设种子,或者每次都在同一毫秒设了相同种子。C++11 之前靠 srand(time(nullptr)),但 time(nullptr) 只精确到秒,连着跑几次程序,很可能种子一样,整个序列就完全重复。
更糟的是:rand() 周期短、分布不均、低位周期更短——用 rand() % 10 取 0–9,结果可能明显偏向某些数。
- 别再用
rand()+srand()做范围随机,尤其不能用%截断 - 用 C++11 的
<random>:真随机、可复现、分布可控 - 种子推荐用
std::random_device,它尝试从系统熵源取真随机数(Linux 下读/dev/urandom)
怎么用 std::uniform_int_distribution 生成 [a, b] 区间的 int?
这是最常用也最容易写错的一环:闭区间 [a, b] 要传入 std::uniform_int_distribution<int>(a, b),不是 (a, b-1) 或 (a, b+1)。底层引擎(如 std::mt19937)输出的是整数,分布器负责映射到指定范围并保证均匀性。
示例:生成 5 到 20(含)之间的随机整数:
立即学习“C++免费学习笔记(深入)”;
std::random_device rd; std::mt19937 gen(rd()); std::uniform_int_distribution<int> dist(5, 20); int x = dist(gen); // x ∈ [5, 20]
- 分布器对象
dist可复用,不要每次生成都重建 - 引擎对象
gen也应复用;重建引擎会丢弃内部状态,还可能重复使用相同种子 - 如果需要可复现结果(比如单元测试),把
rd()换成固定值:std::mt19937 gen(12345);
为什么多线程里用同一个 std::mt19937 实例会出问题?
std::mt19937 不是线程安全的——它的 operator() 会修改内部状态。多个线程同时调用同一个实例,可能破坏状态,导致未定义行为或重复序列。
- 每个线程用独立的引擎实例(推荐):在 thread_local 或线程函数内构造
- 不要全局共享一个
std::mt19937对象 - 如果必须共享,得加锁(性能差,不推荐)
-
std::random_device通常是线程安全的,但标准不强制;生产环境建议也按线程隔离处理
Windows 上 std::random_device 可能返回伪随机数
MSVC 的 std::random_device 在旧版本(如 VS2019 早期)默认回退到 deterministic 算法,rd.entropy() 返回 0,意味着它根本不随机。这会让本该不可预测的种子变成固定值。
- 运行时检查:
if (rd.entropy() == 0) { /* 警告或 fallback */ } - 替代方案:用
rd()多次异或构造种子,或混入std::chrono::high_resolution_clock::now().time_since_epoch().count() - 跨平台稳妥做法:开发阶段用
rd(),CI 或 Windows 生产环境加 fallback 种子逻辑
真正麻烦的从来不是“怎么写一行生成随机数”,而是“怎么确保每次都不一样、线程安全、跨平台一致、还能调试复现”。这些细节漏掉一个,就可能让抽卡概率偏差、测试结果飘忽、甚至安全逻辑被绕过。








