srand函数和rand函数详解

C语言中使用rand函数产生随机数之前总要先调用srand函数,我很困惑为什么要这样做,查完资料之后,才有了这篇博客。

首先我们要知道调用rand()函数产生的值是伪随机值,为什么会是伪随机值呢?这是因为如果我们不初始化随机种子,那么每一次产生的随机数序列都是一样的,看似是随机的,实际上并不随机,所以叫伪随机值,我们可以看看下面的调用。

 

 从上面的调用可以看出来,每次调用得到的序列都是一样的。

rand函数造成相同序列的原因

rand()函数内部是用一个算法先用初始的种子产生一个值作为随机值,下次调用rand函数的时候再用该值作为种子再产生一个值作为下一个随机值,周而复始。

由于初始的种子的值是固定的,所以如果不改变该初始种子的值,那么每次启动程序,调用rand()产生的随机数序列都是一样的。Linux内部的rand()是类似下面的代码实现的:

static unsigned long next = 1;//初始种子的值为1

int my_rand(void) {
    next = next * 1103515245 + 12345;
    return((unsigned)(next/65536) % 32768);
}

void my_srand(unsigned seed) {//通过srand来改变初始种子的值
    next = seed;
}

从以上代码我们可以看到,初始种子是一个静态全局变量,初始值为1,而改变初始种子的值需要调用my_srand函数来进行改变,这就是为什么“不调用srand函数,rand随机出来的序列总是相同”的原因了,因为每次程序重新启动,都会把初始种子的值置为1,那么每次随机出来的序列自然也是相同的了。

通过srand函数和rand函数进行随机值生成

想要rand()函数每次生成的随机序列都是不一样的,那么我们首先就要给srand()的参数中放入一个每时每刻都不一样的数字来作为我们的初始种子,使用时间戳来作为这个数字是再合适不过的了,C语言中时间戳可以由time()这个函数来产生,注意:使用这个函数需要引入头文件<time.h>

函数头文件
int rand(void)<stdlib.h>
void srand(unsigned int seed)<stdlib.h>
time_t time (time_t* timer)<time.h>

 

 从上面的调用可以发现,序列已经不一样了。

srand传参time(NULL)小缺陷

需注意的一点是srand()传参time(NULL)之后,如果在一秒重新调用rand()两次(如上面程序启动两次对rand()进行调用),那么同样会得到两个相同的序列,原因是因为时间戳的值是隔一秒才变一次的,所以如果在一秒内重新调用rand()两次,这个时候时间戳的值是一样的,自然初始化种子的值也是一样的,所以会得到两个相同的序列。 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值