13

我正在尝试使用 C++11 方法生成随机数:

#include <random>
#include <functional>
#include <iostream>

int main(int argc, char *argv[])
{    
    std::normal_distribution<double> normal(0, 1);
    std::mt19937 engine; // Mersenne twister MT19937
    auto generator = std::bind(normal, engine);

    int size = 2;

    engine.seed(0);
    normal.reset();
    for (int i = 0; i < size; ++i)
        std::cout << generator() << std::endl;

    std::cout << std::endl;

    engine.seed(1);
    normal.reset();
    for (int i = 0; i < size; ++i)
        std::cout << generator() << std::endl;

    std::cout << std::endl;

    engine.seed(0);
    normal.reset();
    for (int i = 0; i < size; ++i)
        std::cout << generator() << std::endl;

    return 0;
}

输出是:

0.13453
-0.146382

0.46065
-1.87138

0.163712
-0.214253

这意味着第一个和第三个序列并不相同,即使它们以相同的编号作为种子。请问,我做错了什么?是个

std::normal_distribution<double>

只是数学意义上的一个函数(确定性地从 x 中产生 y)还是我遗漏了什么?如果只是一个函数,那么reset方法实际上是做什么的呢?

4

2 回答 2

19

您正在绑定引擎和分发,因此以下重置调用不会影响绑定功能。

解决方案是将引用绑定到引擎和发行版

auto generator = std::bind(std::ref(normal), std::ref(engine));
于 2012-06-12T06:47:28.053 回答
11

您遇到的问题是std::bind. std::bind复制其论点。之所以std::bind进行复制,是因为将来当参数可能不再存在时,该函数将在某个未知时间点被调用。engine.seed()这意味着您对等的调用是无用的。使用std::ref您可以通过引用绑定您的参数,这将为您提供预期的输出。

auto generator = std::bind(std::ref(normal), std::ref(engine));
于 2012-06-12T06:48:09.070 回答