Chinaunix首页 | 论坛 | 博客
  • 博客访问: 2857339
  • 博文数量: 471
  • 博客积分: 7081
  • 博客等级: 少将
  • 技术积分: 5369
  • 用 户 组: 普通用户
  • 注册时间: 2012-01-04 21:55
文章分类

全部博文(471)

文章存档

2014年(90)

2013年(69)

2012年(312)

分类: C/C++

2012-09-24 11:35:13

来自:http://atell.iteye.com/blog/1181907
题目:

给定一个函数rand5(),该函数可以随机生成1-5的整数,且生成概率一样。现要求使用该函数构造函数rand7(),使函数rand7()可以随机等概率的生成1-7的整数。

思路:

很多人的第一反应是利用rand5() + rand()%3来实现rand7()函数,这个方法确实可以产生1-7之间的随机数,但是仔细想想可以发现数字生成的概率是不相等的。rand()%3 产生0的概率是1/5,而产生1和2的概率都是2/5,所以这个方法产生6和7的概率大于产生5的概率。

正确的方法是利用rand5()函数生成1-25之间的数字,然后将其中的1-21映射成1-7,丢弃22-25。例如生成(1,1),(1,2),(1,3),则看成rand7()中的1,如果出现剩下的4种,则丢弃重新生成。

 此题的关键是让生成的 1 到 7 的数出现概率相同,两个随机函数运行的值不同,才有0 5 10 15 20 分别与 
个位1 2 3 4 5 组合1-25这25个数随机产生的,映射到21 也是随机产生的

简单实现:

Java代码 复制代码 收藏代码
  1. public class Test {     
  2.     public int rand7() {     
  3.         int x = 22;     
  4.         while(x > 21) {     
  5.             x = rand5() + (rand5() - 1)*5;     
  6.         }     
  7.         return 1 + x%7;     
  8.     }     
  9.      
  10. }    
public class Test { public int rand7() { int x = 22; while(x > 21) { x = rand5() + (rand5() - 1)*5; } return 1 + x%7; } }

   我的备注:

    这种思想是基于,rand()产生[0,N-1],把rand()视为N进制的一位数产生器,那么可以使用rand()*N+rand()来产生2位的N进制数,以此类推,可以产生3位,4位,5位...的N进制数。这种按构造N进制数的方式生成的随机数,必定能保证随机,而相反,借助其他方式来使用rand()产生随机数(如 rand5() + rand()%3 )都是不能保证概率平均的。

    此题中N为5,因此可以使用rand5()*5+rand5()来产生2位的5进制数,范围就是1到25。再去掉22-25,剩余的除3,以此作为rand7()的产生器.

阅读(700) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~