zoukankan      html  css  js  c++  java
  • 线性同余法生成为随机数

    线性同余方法(LCG)是个产生伪随机数的方法。

    它是根据递归公式:

    N_{j+1} equiv (A 	imes N_j + B ) pmod{M}

    其中A,B,M是产生器设定的常数。

    LCG的周期最大为M,但大部分情况都会少于M。要令LCG达到最大周期,应符合以下条件:

    1. B,M互质
    2. M的所有质因子的能整除A-1
    3. M是4的倍数A-1也是;
    4. A,B,N_0都比M小;
    5. A,B是正整数。

        线性同余算法有m 、a 、c 和X0 4个参数,通过置Xn + 1 ≡aXn + c (mod m) ,求得随机数序列< Xn > , 这个序列称作线性同余序列。m、a 、c 和X0 分别称做模数、乘数、增量和初始值。线性同余方法速度快,如果对乘数和模数进行适当的选择,可以满足用于评价一个随机数产生器的3 种准则:
    1.这个函数应该是一个完整周期的产生函数。也就是说,这个函数应该在重复之前产生出0 到m之间的所有数;
    2.产生的序列应该看起来是随机的;
    3.这个函数应该用32bit 算术高效实现。

    在我的算法中,a=7^5;c=0;m=2^31-1; x0为系统时间;

    我的代码如下:

    [cpp] view plain copy
     
    1. #include <stdio.h>  
    2. #include <time.h>  
    3. static unsigned long rand_seed;  
    4. void mysrand (unsigned long int);  
    5. void myrand ();  
    6. int  
    7. main (void)  
    8. {  
    9.     int i;  
    10.   
    11.     mysrand (time (NULL));  
    12.     for (i = 0; i < 100; i++)  
    13.       {  
    14.           myrand ();  
    15.       }  
    16.     return 0;  
    17. }  
    18.   
    19. void  
    20. mysrand (unsigned long seed)  
    21. {  
    22.     rand_seed = seed;  
    23. }  
    24.   
    25. void  
    26. myrand ()  
    27. {  
    28.     rand_seed = (rand_seed * 16807L) % ((1 << 31) - 1);  
    29.     printf ("%ld ", rand_seed);  
    30. }  

    ===============PS==================

    产生随机种子的方法很多,目前用得比较多的是使用系统时间为种子。我觉得这种方法也不妥当。假如我批量执行程序,程序执行的时间是几个ms,那么几个相邻程序的种子就是一样的,产生的结果因此也是一样的。(因为系统时间是按照秒来计算的,一秒内执行多少次,产生的随机种子就有多少相同的。)

    比较流行的是使用seed = (seed * 9301 + 49297) % 233280;
    return seed / (233280.0);

  • 相关阅读:
    设计模式 : Template method 模板方法模式 -- 行为型
    设计模式:Strategy 策略模式 -- 行为型
    没有评审的技术都是扯淡
    TCP/IP 协议大致的概念
    HTTP 2.0 与 tomcat
    相对URL拼接为绝对URL的过程
    URI与URL区别
    《淘宝首页性能优化实践》文章阅读
    第一章 Linux内核简介
    vim中执行shell命令小结
  • 原文地址:https://www.cnblogs.com/jingxuan-li/p/8278361.html
Copyright © 2011-2022 走看看