线程安全的并行RNG比顺序rand()慢
Thread-safe parallel RNG slower than sequential rand()
我使用这个带有线程安全函数的Pi计算版本
rand_r
但是,与使用
的顺序程序相比,并行运行该程序似乎更慢(答案是错误的)。rand()
不是线程安全的。似乎这种使用方式也不是线程安全的。但是我不明白为什么,因为我读了很多关于线程安全prng的问题,并且了解到rand_r应该足够安全。
#include <iostream>
#include <random>
#include <ctime>
#include "omp.h"
#include <stdlib.h>
using namespace std;
unsigned seed;
int main()
{
double start = time(0);
int i, n, N;
double x, y;
N = 1<<30;
n = 0;
double pi;
#pragma omp threadprivate(seed)
#pragma omp parallel private(x, y) reduction(+:n)
{
for (i = 0; i < N; i++) {
seed = 25234 + 17 * omp_get_thread_num();
x = rand_r(&seed) / (double) RAND_MAX;
y = rand_r(&seed) / (double) RAND_MAX;
if (x*x + y*y <= 1)
n++;
}
}
pi = 4. * n / (double) (N);
cout << pi << endl;
double stop = time(0);
cout << (stop - start) << endl;
return 0;
}
注:顺便问一下,
中的神奇数字是什么?seed = 25234 + 17 * omp_get_thread_num();
?我从一些答案中偷来的。
EDIT: Gilles的评论帮助了我。决议是:1. 切换for循环和种子初始化的行。2. 为
添加#pragma omp修改后的代码为
#pragma omp parallel private(x, y, seed)
{
seed = 25234 + 17 * omp_get_thread_num();
#pragma omp for reduction(+:n)
for (int i = 0; i < N; i++) {
x = (double) rand_r(&seed) / (double) RAND_MAX;
y = (double) rand_r(&seed) / (double) RAND_MAX;
if (x*x + y*y <= 1)
n++;
}
}
问题解决
显然,与rand()相比,rand_r()中的指令更多。下面是从一个实现中复制的。因此,rand_r()比rand()需要更多的时间来完成一轮是合理的。
int
rand_r(unsigned int *ctx)
{
u_long val = (u_long) *ctx;
int r = do_rand(&val);
*ctx = (unsigned int) val;
return (r);
}
static u_long next = 1;
int
rand()
{
return (do_rand(&next));
}
由于rand()不是线程安全的,如果并行使用rand(),输出可能是不正确的。更糟糕的是,你仍然会得到一个结果,但不知道它在小规模测试中是否正确。
相关文章:
- CMake-按正确顺序将项目与C运行时对象文件链接
- 函数调用中参数的顺序重要吗
- 为什么不;名字在地图上是按顺序排列的吗
- 将Integer转换为4字节的unsined字符矢量(按大端字节顺序)
- 数到第n个楼梯的路(顺序无关紧要)
- 优先顺序:智能指针和类析构函数
- 在循环中按顺序遍历成员变量
- 独立读取-修改-写入顺序
- QML按钮点击功能执行顺序
- 为什么我不能将 rand() 与数组的大小一起使用?
- C++中数据类型修饰符的顺序
- 当比特(而不是字节)的顺序至关重要时的持久性
- C++从其他 constexpr 创建 lambda 不能按顺序执行 Constexpr
- 通过选项卡的文本设置QTabWidget顺序
- c++11评估顺序(未定义的行为)
- 如何在C++中递归地按相反顺序打印集合
- 为什么新的随机库比std::rand()更好
- 给定顺序中的事件处理
- 具有包含其他对象的类的对象创建顺序
- 线程安全的并行RNG比顺序rand()慢