php专区

 首页 > php专区 > PHP进阶 > 算法 > 如何用随机函数rand5来构造随机函数rand7

如何用随机函数rand5来构造随机函数rand7

分享到:
【字体:
导读:
         摘要:rand5()它能够等概率生成1-5之间的整数。所谓等概率就是1,2,3,4,5生产的概率均为0.2。现在利用rand5(),构造一个能够等概率生成1-7的方法。这里有两个特别重要的点,一是如果rand5()+rand5(),我们能够产生一个均匀分布的1-10吗?答案是否定的。比如对于6来讲(4+2,2+4,3+3),...

如何用随机函数rand5来构造随机函数rand7

试一下以对话的方式写博~如果看不到人物头像,请刷新页面获取最新的CSS。如果有建议或意见,欢迎到我的微博上跟帖~ 这里是腾讯微博,这里是新浪微博。

常规方法

  • 今天公司有一个面试题是这样的:假如有一个函数rand5能等概率生成1 - 5 之间的整数,如何利用rand5来实现rand7?rand7函数的要求是能够等概率生成1 - 7之间的整数。说实话我自己也不是很清楚。
  • 这个问题很经典的。carreercup那本书上有个常见的解法,我记得算法大概是这样的,用PHP写写吧:
echo 'rand7 = '.rand7();

function rand7()
{
	while (true)
	{
		//得出[0,24]的平均分布
		$i = 5 * (rand5() - 1) + (rand5() - 1);
		//只取前21个, 前21个也是平均分布,然后mod 7
		if( $i 

  • 这么写是什么意思呢?
  • 两次使用rand5(),可以生成1-25的所有数。5 * (rand5() - 1) 可以生成 0 - 20,而后面的则可以生成0 - 4。用数学表达的话就是 [0, 24]。
  • [0, 24] 范围内生成的随机数$i如果大于21,就用 while (true) 重新生成。当$i
  • 明白了。当 $i

    算法的一些释疑

    • 话说得出[0,24]的平均分布的 $i = 5 * (rand5() - 1) + (rand5() - 1); 为什么要这么写呢?为什么不能直接 $i = 6 * (rand5() - 1) ?
    • rand5()产生的是[1,5]的均匀分布,但是有没有发现最后产生的rand7()却是[0,6]的,那么平均分布是如何实现的呢?
    • 5*(rand5()-1) 生成的是 0, 5,10,15,20各20%的概率,rand5()-1 是0,1,2,3,4各20%概率,两者相加,就是0-24各1/25的概率,这样就保证了平均分布的问题了。基本的概率和排列组合问题了。
    • 前21个也是平均分布,那么我取 $i
    • 应该也可以,但是增加了可能的循环次数。你看到 while (true) 这行代码吗?当while里面的条件不满足,循环就会一直下去。所以从程序上考虑,就是用了21而非14吧。
    • 晚些时候

      • 我Google了下,貌似还有其它思路。先用2个rand5产生rand10(注意,不是相加),然后从rand10产生rand7。
      // Gen 0, 1 equal probability
      int rand01()
      {
          int i = rand5();
          while (i > 4) {i = rand5();}
          return i % 2;
      }
       
      // Gen 0, 1, 2, 3, 4, 5, 6, 7 equal probability
      int rand07()
      {
          return rand01() << 2 + rand01() << 1 + rand01();
      }
       
      // Gen 1, 2, 3, 4, 5, 6, 7 equal probability
      int rand7()
      {
          int i = rand07();
          while (i == 0) {i = rand07();}
          return i;
      }
      
      • 还有一种方法,可以直接把概率问题转化到矩阵中解决。
      int matrix[5][5];
       
      memset(matrix, 0, sizeof(matrix));
       
      // Set matrix with num 1-7, each num has the same count.
      for (int i = 1; i 
      
      
      • 你还真好学。。

      通过这个面试题学到了等概率问题的各种解法,可以从把数从二进制角度看,可以用公式拼接出更大的等概率值域空间,也可以直接把概率问题转化到矩阵中解决。

      rand5() 它能够等概率生成 1-5 之间的整数。所谓等概率就是1,2,3,4,5 生产的概率均为 0.2 。现在利用rand5(), 构造一个能够等概率生成 1- 7 的方法。 这里有两个特别重要的点,一是 如果 rand5() + rand5(), 我们能够产生一个均匀分布的 1 - 10 吗? 答案是否定的。比如对于 6来讲(4+2, 2+4, 3+3),它被生成的生成的概率比1 (1+0,0+1)要大。

      第二个点就是我们不可能用rand5()直接产生 1- 7 的数,不管你用加减乘除都不行。所以,我们要构造一个更大的范围,使得范围里每一个值被生成的概率是一样的,而且这个范围是7的倍数。

      先产生一个均匀分布的 0, 5, 10, 15, 20的数,再产生一个均匀分布的 0, 1, 2, 3, 4 的数。相加以后,会产生一个 0到24的数,而且每个数(除0外)生成的概率是一样的。我们只取 1 - 21 这一段,和7 取余以后+1就能得到完全均匀分布的1-7的随机数了。

      本文地址:http://www.nowamagic.net/librarys/veda/detail/2172,欢迎访问原出处。

    如何用随机函数rand5来构造随机函数rand7

  • 打酱油

  • 震惊

  • 呵呵


  • 鄙视
标签标签:PHP ,CSS数据库 ,优化
立即登录 | 注册新用户
最新评论
  • 不吐不快,赶紧来一发!
栏目推荐
  • 常见PHP排序算法的复习和总结
  • 谈谈PHP树生成迷宫及A*自动寻路算法
  • PHP SPL标准库之数据结构栈(SplStack)
  • 一个Python实现的精简版遗传算法
  • 余弦相似度实现—php数据分析 PHP实现
  • 产生n位元的所有格雷码
  • 十进制码转格雷码的PHP的一种实现写法
  • 一个php写的截取字符串的函数
  • php计算一个页面执行时间函数写法
  • 分享一个用php写的人民币数字转人民币大写的函数
关注 第一PHP社区 微信公众号
热门点击
  • 新闻APP后端系统架构成长之路 - 高可用架构设计
  • PHP基础: CLI模式开发不需要任何一种Web服务器
  • 基础教程:svn命令在linux下的使用
  • 亿级Web系统搭建——单机到分布式集群
  • HTTP简介,http是一个属于应用层的面向对象的协议
  • Serverless技术架构,传说中的FAAS(Function as a Service),极简运维,无限扩容
  • PHP漏洞全解(六)跨网站请求伪造
  • php 获取今日、昨日、上周、本月的起始时间戳和结束时间戳的方法
  • PHP学习路线以及10个PHP优化技巧
  • 适用于PHP初学者的学习线路和建议
本站推荐
  • 2016PHP技术盛会:如何编写可测试的代码
  • MySQL改密码报错:ERROR 1045: Access denied for user: 'root@localhost' (Using password: YES)
  • phpmyadmin导入导出数据库文件最大限制的解决方法
  • Linux环境下MySQL数据库大小写区分问题
  • php 5.4中php-fpm 的重启、终止操作命令
  • Key/Value之王Memcached初探:三、Memcached解决Session的分布式存储场景的应用
  • Key/Value之王Memcached初探:二、Memcached在.Net中的基本操作
  • Key/Value之王Memcached初探:一、掀起Memcached的盖头来
  • NoSQL初探之人人都爱Redis:(4)Redis主从复制架构初步探索
  • NoSQL初探之人人都爱Redis:(3)使用Redis作为消息队列服务场景应用案例
分享到:
漫谈递归:递归的思想
漫谈递归:递归的思想 为什么要用递归 编程里面估计最让人摸不着头脑的基本算法就是递归了。很多时候我们看明白一个复杂的递归都有点费时间,尤其对模型所描述的问题概念不清的时候,想要自己设计一个递归那么就更是有难度了。 很多不理解递归的人(今天在csdn里面看到一个初学者的留言),总认为递归完全没必要,...
海盗分宝石面试题的头脑风暴
海盗分宝石面试题的头脑风暴 有个面试题,是这样的: 五个海盗得到100颗钻石,颗颗价值连城。这五个海盗非常聪明,都想自己得到钻石最多。因而他们设计了个规则,根据抽签后的顺序, 每个人提出个分配方案,如果有半数以上(不包括半数)的人表决通过,则按这个方案执行,否则提出方案的人要被扔到海里喂鱼。下一个...
  •         php迷,一个php技术的分享社区,专属您自己的技术摘抄本、收藏夹。
  • 在这里……