C语言rand函数产生随机数详解

  • 1.rand函数的原型,功能和界限
  • 2.用rand函数获取随机数
  • 3.随机数生成原理

1.rand函数的原型,功能和界限

rand函数的原型:

#include int rand(void);

rand函数的功能: 用来产生一个随机数

rand函数的界限: stdlib.h头文件中有宏#define RAND_MAX 0x7fff,这表示rand产生一个0-0x7fff的随机数,即最大是32767的一个数

2.用rand函数获取随机数

rand函数获取随机数的基础用法为:

rand()%a;

a是一个整数,返回值为0到a的随机数,包含0,但不包含a。

如果想要产生一个0-99之间的随机数,那么用法如下:

int num = rand() % 100;

如果想要产生一个1-100之间的随机数,那么用法如下:

int num = rand() % 100 + 1;

同理,如果想要产生一个1-60之间的随机数,那么用法如下:

 int num = rand() % 60 + 1;//产生一个1-60之间的数

从上面的例子中我们可以推理出m—–n的随机数:

rand()%(n-m+1) + m;

比如获取15~43的随机数:

rand()%29+15;

获取小数:
我们只需要将返回值除以10或者其他合适的数,并且在输出时保留若干位小数就可以获取随机小数。

这里一定要用10.0,不能用整型。

3.随机数生成原理

我们先来看看下面这段代码,反复运行该段代码,我们会发现每次产生的随机数的值都是相同的。

#include#includeint main(){int ch = rand();printf("%d\n", ch);return 0;}

这是因为rand函数产生的随机数是伪随机数,它除了被a的值控制外,还被一个隐藏的全局变量seed控制。它实际上是根据a和一个种子按照某个公式推算出来的。

在调用rand函数之前,会先查询是否主动调用过srand(seed)来为伪随机数生成器设定种子,如果有,那么就按照我们的代码设定种子,即初始化seed的起始值,若没有调用srand(seed),那么系统会自动给seed赋初始值,即自动调用srand(1)一次,也就是将seed的值设置为1。如果每次调用rand函数时,种子相同,那么根据公式产生的伪随机数将是相同的。

上面那段代码因为每次运行程序你都没有主动调用srand函数给seed赋值,因此每次都会默认调用srand(1),将seed的值赋值为1,也就是每次seed的值都相同,所以产生的伪随机数不会发生变化。

所以为了让每次产生的随机数不相同,那么我们就需要让每次调用rand函数之前设定的种子不相同,我们可以使用系统时间作种子srand(time(NULL);由于每次运行程序的时间肯定不相同,那么每次设定的种子也就不相同,那么产生的随机数肯定也就不一样了。

srand函数的原型:

#includevoid srand( unsigned int seed );

我们使用time函数来获取系统时间,time函数的原型如下:

#includetime_t time(time_t *tloc);//time_t类型被定义为一个长整型

使用time函数得到的值是一个时间戳,即从1970年1月1日0点到现在时间的秒数,然后将得到的time_t类型数据转化为(unsigned int)的数,再传给srand函数,用法如下:

int main(){srand((unsigned int)time(NULL));int ch = rand();printf("%d\n", ch);return 0;}

添加了srand((unsigned int)time(NULL));之后,当你再次运行该段代码时,你就会发现,每次生成的随机数就不同了。

如果仍然觉得时间间隔太小,可以在(unsigned int)time(0)或者(unsigned int)time(NULL)后面乘上某个合适的整数。 例如,srand((unsigned int)time(NULL)*10),time的参数传NULL表示不需要经过参数得到time_t数据。

还有另外一种初始化种子的方式如下,用进程的pid作为种子值seed,在同一个程序中,这样的种子的值是相同的。

srand((unsigned int)getpid());