#include <sys/time.h>
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <inttypes.h>
#include <stdint.h>
#include <limits.h>
unsigned long static openwrong=0, readwrong=0,goodnum=0,overflow=0;
unsigned long myrandom()
{
unsigned long data;
struct timeval ti;
gettimeofday(&ti,NULL);
unsigned long seed=ti.tv_usec;
int fd=open("/dev/random",O_RDONLY);//1022 次以后出问题,fd<=0;
if(fd<0 )
{
srandom(seed);
data=random();
openwrong++;
}
else if (read(fd,&data,sizeof(data))<=0)
{
srandom(seed);
data=random();
readwrong++;
}
else if(data>SSIZE_MAX)// 有可能溢出
{
overflow++;
data=data%SSIZE_MAX;
}
else goodnum++;
return data;
}
int main(int argc, char **argv)
{
unsigned long i,num=1050;//1022 次以后出问题
for(i=1;i<num;i++)
printf("myrandom%ld=%ld\n",i, myrandom());
printf("openwrong=%ld,readwrong=%ld,overflow=%ld,goodnum=%ld\n",openwrong,readwrong,overflow,goodnum);
return 0;
}
// 看到一本老外写的书。上面写了利用 /dev/random 和 /dev/urandom 计算伪随机数较为可信。
// 可是我试了,只可利用 1021 次。为何?
Linux 下面一个程序能打开的最大文件描述符个数可以通过以下命令查看:
ulimit -n
一般默认的是 1024 个,一个程序运行时会打开标准输入、标准输出和标准错误 3 个文件描述符,看一下你的 open 函数
int fd=open("/dev/random",O_RDONLY);
由于你没有调用 close 函数关闭 fd,每运行一次 open 函数就会打开一个文件描述符,当运行到第 1021 次的时候,程序已经打开了 1024 个文件描述符,再运行第 1022 次的时候 open 函数就会打开错误。解决方法就是关闭已经打开的文件描述符,在 myrandom 函数中 return data 的前面加上下面一行代码:
close(fd);