C 语言

关注公众号 jb51net

关闭
首页 > 软件编程 > C 语言 > C++随机数生成

C++随机数生成工具实现详解

作者:南城花随雪。

这篇文章主要介绍了C++随机数生成工具的实现,随机数是计算机科学、数值计算与工程应用中最基础、也最关键的组件之一,需要的朋友可以参考下

一、项目背景详细介绍

随机数生成器(Random Number Generator,简称 RNG)是计算机科学、数值计算与工程应用中最基础、也最关键的组件之一。从最底层的系统软件,到高层的算法与模型,几乎所有领域都离不开随机数:

在 C++ 中,虽然标准库已经提供了 <random>,但在以下场景中,我们必须自己实现随机数生成器:

因此,掌握 RNG 的原理并亲手实现一个高质量随机数生成器,是数值计算工程师的基本功。

二、项目需求详细介绍

2.1 功能性需求

本项目目标是:
👉 在 C++ 中从零实现一个可扩展、可复现、可教学的随机数生成框架

具体要求如下:

实现一个核心随机数生成器(Uniform RNG)

支持设置随机种子(Seed)

支持生成:

所有分布基于同一 RNG 内核

提供清晰、稳定的接口设计

2.2 非功能性需求

2.3 适用场景

三、相关技术详细介绍

3.1 为什么不能直接用rand()

C 标准库的 rand() 存在多个问题:

因此,在工程与科研中几乎从不推荐使用 rand()

3.2 常见随机数生成算法对比

算法周期优点缺点
LCG简单低质量
Mersenne Twister极长高质量复杂
Xorshift线性相关
PCG快 + 高质量稍复杂

📌 本项目选择:Xorshift + 教学友好设计

3.3 Xorshift 算法原理

Xorshift 是 George Marsaglia 提出的一类随机数生成算法,其核心思想是:

使用异或(XOR)与位移(Shift)操作构造长周期随机序列

以 Xorshift64 为例:

x ^= x << a x ^= x >> b x ^= x << c reminder

特点:

四、实现思路详细介绍

4.1 架构设计

RandomEngine ├─ nextUint64() → 核心随机数 ├─ uniform01() → [0,1) ├─ uniform(a,b) → 均匀分布 ├─ normal(mean,σ) → 正态分布 └─ exponential(λ) → 指数分布 

4.2 分布生成策略

4.3 可复现性设计

所有随机数只依赖:

不使用系统时间作为默认种子

五、完整实现代码

/******************************************************
 * File: random_engine.h
 * Description: 随机数生成器接口
 ******************************************************/
#ifndef RANDOM_ENGINE_H
#define RANDOM_ENGINE_H
#include <cstdint>
class RandomEngine
{
public:
    explicit RandomEngine(uint64_t seed = 88172645463325252ull);
    uint64_t nextUint64();
    double uniform01();
    double uniform(double a, double b);
    double normal(double mean = 0.0, double stddev = 1.0);
    double exponential(double lambda);
private:
    uint64_t state;
    bool hasSpare;
    double spare;
};
#endif
/******************************************************
 * File: random_engine.cpp
 * Description: Xorshift 随机数生成器实现
 ******************************************************/
#include "random_engine.h"
#include <cmath>
#include <stdexcept>
/* 构造函数:初始化种子 */
RandomEngine::RandomEngine(uint64_t seed)
    : state(seed), hasSpare(false)
{
    if (state == 0)
        state = 88172645463325252ull;
}
/* 核心 Xorshift64 算法 */
uint64_t RandomEngine::nextUint64()
{
    uint64_t x = state;
    x ^= x << 13;
    x ^= x >> 7;
    x ^= x << 17;
    state = x;
    return x;
}
/* 生成 [0,1) 上均匀分布 */
double RandomEngine::uniform01()
{
    return (nextUint64() >> 11) * (1.0 / 9007199254740992.0);
}
/* 生成 [a,b) 上均匀分布 */
double RandomEngine::uniform(double a, double b)
{
    if (a >= b)
        throw std::invalid_argument("uniform: a must be < b");
    return a + (b - a) * uniform01();
}
/* 正态分布:Box–Muller 变换 */
double RandomEngine::normal(double mean, double stddev)
{
    if (stddev <= 0.0)
        throw std::invalid_argument("normal: stddev must be positive");
    if (hasSpare)
    {
        hasSpare = false;
        return mean + stddev * spare;
    }
    double u, v, s;
    do
    {
        u = uniform(-1.0, 1.0);
        v = uniform(-1.0, 1.0);
        s = u * u + v * v;
    } while (s >= 1.0 || s == 0.0);
    s = std::sqrt(-2.0 * std::log(s) / s);
    spare = v * s;
    hasSpare = true;
    return mean + stddev * (u * s);
}
/* 指数分布 */
double RandomEngine::exponential(double lambda)
{
    if (lambda <= 0.0)
        throw std::invalid_argument("exponential: lambda must be positive");
    return -std::log(1.0 - uniform01()) / lambda;
}
/******************************************************
 * File: main.cpp
 * Description: 示例与测试
 ******************************************************/
#include <iostream>
#include "random_engine.h"
int main()
{
    RandomEngine rng(12345);
    std::cout << "Uniform [0,1):\n";
    for (int i = 0; i < 5; ++i)
        std::cout << rng.uniform01() << std::endl;
    std::cout << "\nNormal(0,1):\n";
    for (int i = 0; i < 5; ++i)
        std::cout << rng.normal() << std::endl;
    std::cout << "\nExponential(lambda=2):\n";
    for (int i = 0; i < 5; ++i)
        std::cout << rng.exponential(2.0) << std::endl;
    return 0;
}

六、代码详细解读(仅解读方法作用)

6.1 nextUint64

6.2 uniform01

6.3 normal

6.4 exponential

七、项目详细总结

通过本项目,我们:

该随机数生成器:

八、项目常见问题及解答

Q1:是否适合密码学?

A:不适合,需要使用 CSPRNG(如 AES-CTR)。

Q2:是否线程安全?

A:当前版本不是,可通过线程私有实例解决。

Q3:周期有多长?

A:Xorshift64 周期为 264−12^{64} - 1264−1。

九、扩展方向与性能优化

以上就是C++随机数生成工具实现详解的详细内容,更多关于C++随机数生成的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文