How to generate random numbers in parallel?

后端 未结 6 754
慢半拍i
慢半拍i 2020-11-30 07:04

I want to generate pseudorandom numbers in parallel using openMP, something like this:

int i;
#pragma omp parallel for
for (i=0;i<100;i++)
{
    printf(\"         


        
相关标签:
6条回答
  • 2020-11-30 07:21

    Random numbers can be generated very fast,so usually the memory would be the bottleneck. By dividing this task between several threads you create additional communication and syncronization overheads (and sinchronization of caches of different cores is not cheap).

    It would be better to use a single thread with a better random() function.

    0 讨论(0)
  • 2020-11-30 07:24

    You cannot use the C rand() function from multiple threads; this results in undefined behavior. Some implementations might give you locking (which will make it slow); others might allow threads to clobber each other's state, possibly crashing your program or just giving "bad" random numbers.

    To solve the problem, either write your own PRNG implementation or use an existing one that allows the caller to store and pass the state to the PRNG iterator function.

    0 讨论(0)
  • 2020-11-30 07:25

    I'll post here what I posted to Concurrent random number generation :

    I think you're looking for rand_r(), which explicitly takes the current RNG state as a parameter. Then each thread should have it's own copy of seed data (whether you want each thread to start off with the same seed or different ones depends on what you're doing, here you want them to be different or you'd get the same row again and again). There's some discussion of rand_r() and thread-safety here: whether rand_r is real thread safe? .

    So say you wanted each thread to have its seed start off with its thread number (which is probably not what you want, as it would give the same results every time you ran with the same number of threads, but just as an example):

    #pragma omp parallel default(none)
    {
        int i;
        unsigned int myseed = omp_get_thread_num();
        #pragma omp for
        for(i=0; i<100; i++)
                printf("%d %d %d\n",i,omp_get_thread_num(),rand_r(&myseed));
    }
    

    Edit: Just on a lark, checked to see if the above would get any speedup. Full code was

    #define NRANDS 1000000
    int main(int argc, char **argv) {
    
        struct timeval t;
        int a[NRANDS];
    
        tick(&t);
        #pragma omp parallel default(none) shared(a)
        {
            int i;
            unsigned int myseed = omp_get_thread_num();
            #pragma omp for
            for(i=0; i<NRANDS; i++)
                    a[i] = rand_r(&myseed);
        }
        double sum = 0.;
        double time=tock(&t);
        for (long int i=0; i<NRANDS; i++) {
            sum += a[i];
        }
        printf("Time = %lf, sum = %lf\n", time, sum);
    
        return 0;
    }
    

    where tick and tock are just wrappers to gettimeofday(), and tock() returns the difference in seconds. Sum is printed just to make sure that nothing gets optimized away, and to demonstrate a small point; you will get different numbers with different numbers of threads because each thread gets its own threadnum as a seed; if you run the same code again and again with the same number of threads you'll get the same sum, for the same reason. Anyway, timing (running on a 8-core nehalem box with no other users):

    $ export OMP_NUM_THREADS=1
    $ ./rand
    Time = 0.008639, sum = 1074808568711883.000000
    
    $ export OMP_NUM_THREADS=2
    $ ./rand
    Time = 0.006274, sum = 1074093295878604.000000
    
    $ export OMP_NUM_THREADS=4
    $ ./rand
    Time = 0.005335, sum = 1073422298606608.000000
    
    $ export OMP_NUM_THREADS=8
    $ ./rand
    Time = 0.004163, sum = 1073971133482410.000000
    

    So speedup, if not great; as @ruslik points out, this is not really a compute-intensive process, and other issues like memory bandwidth start playing a role. Thus, only a shade over 2x speedup on 8 cores.

    0 讨论(0)
  • 2020-11-30 07:35

    On linux/unix you can use

    long jrand48(unsigned short xsubi[3]);
    

    where xsubi[3] encodes the state of the random number generator, like this:

    #include<stdio.h>
    #include<stdlib.h>
    #include <algorithm> 
    int main() {
      unsigned short *xsub;
    #pragma omp parallel private(xsub)
      {  
        xsub = new unsigned short[3];
        xsub[0]=xsub[1]=xsub[2]= 3+omp_get_thread_num();
        int j;
    #pragma omp for
        for(j=0;j<10;j++) 
          printf("%d [%d] %ld\n", j, omp_get_thread_num(), jrand48(xsub));
      }
    }
    

    compile with

    g++-mp-4.4 -Wall -Wextra -O2 -march=native -fopenmp -D_GLIBCXX_PARALLEL jrand.cc -o jrand
    

    (replace g++-mp-4.4 with whatever you need to call g++ version 4.4 or 4.3) and you get

    $ ./jrand 
    0 [0] 1344229389
    1 [0] 1845350537
    2 [0] 229759373
    3 [0] 1219688060
    4 [0] -553792943
    5 [1] 360650087
    6 [1] -404254894
    7 [1] 1678400333
    8 [1] 1373359290
    9 [1] 171280263
    

    i.e. 10 different pseudorandom numbers without any mutex locking or race conditions.

    0 讨论(0)
  • 2020-11-30 07:36

    It seems like that rand has a global shared state between all threads on Linux and a thread local storage state for it on Windows. The shared state on Linux is causing your slowdowns because of the necessary synchronization.

    I don't think there is a portable way in the C library to use the RNG parallel on multiple threads, so you need another one. You could use a Mersenne Twister. As marcog said you need to initialize the seed for each thread differently.

    0 讨论(0)
  • 2020-11-30 07:40

    Get each thread to set a different seed based on its thread id, e.g. srand(omp_get_thread_num() * 1000);

    0 讨论(0)
提交回复
热议问题