关于/dev/random和/dev/urandom求随机数

#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);