每日算法系列【LeetCode 470】用 Rand7() 实现 Rand10()

简介: 已有方法 rand7 可生成 1 到 7 范围内的均匀随机整数,试写一个方法 rand10 生成 1 到 10 范围内的均匀随机整数。不要使用系统的 Math.random() 方法。

题目描述


已有方法 rand7 可生成 1 到 7 范围内的均匀随机整数,试写一个方法 rand10 生成 1 到 10 范围内的均匀随机整数。

不要使用系统的 Math.random() 方法。

思考

  • rand7()调用次数的 期望值 是多少 ?
  • 你能否尽量少调用 rand7() ?

题解


刚看到这题觉得挺有意思的,再看一脸懵逼,这怎么做?后来看了题解才懂了,原来是这个意思。

题目要求只能给你用 rand7 函数,也就是均匀生成 1 到 7 之间的整数。但是现在要求你生成 1 到 10 之间的整数,那么肯定只生成一次是不够的,因为状态数都不够嘛,那就生成多次看看。

如果生成两次,那么就得到了两个 1 到 7 之间的整数,然后怎么转换为 1 到 10 呢。如果这两个数两两组合,那么可以得到 49 种状态,可以用来表示 1 到 49 这 49 个数字,如果想要让 1 到 10 均匀分布,那么每个数字最多只能分配 4 次。具体分配情况如下所示:

1  2  3  4  5  6  78  9  10 1  2  3  45  6  7  8  9  10 12  3  4  5  6  7  89  10 1  2  3  4  56  7  8  9  10 .  ..  .  .  .  .  .  .


image.png

所以平均只需要 2.45 次就可以均匀的采样到 1 到 10 之间的整数啦。那么这背后的数学原理是什么呢?其实就是拒绝采样

蒙特卡洛方法大家应该都很熟悉了,就是采样来求分布,比如求一个直径为 1 的圆的概率,我们可以用一个边长为 1 的正方形包住它,然后随机往里面扔豆子,扔 10000 个,看最后有多少落在了圆里面,那么除以 10000 就是圆的面积了。


image.png

代码


c++

// The rand7() API is already defined for you.
   // int rand7();
   // @return a random integer in the range 1 to 7
class Solution {public:
    int rand10() {
    int r1, r2, num;
    do {
    r1 = rand7(); 
    r2 = rand7();
    num = (r1 - 1) * 7 + r2; 
    } while (num > 40);
    return (num - 1) % 10 + 1;    }
 };

后记

这题题目虽简单,背后的思想还是很有意思的,拒绝采样可以用在深度学习中的很多应用场景里,特别是你的分布很难进行采样的时候,就可以用拒绝采样来模拟。

当然这题还有其他采样方法可以缩小期望采样次数,比如如何利用这 9 个被拒绝的点呢?留给大家思考(其实是我懒得写了)。

image.png

作者简介:godweiyang知乎同名华东师范大学计算机系硕士在读,方向自然语言处理与深度学习喜欢与人分享技术与知识,期待与你的进一步交流~


相关文章
|
程序员
【Leetcode】225. 用队列实现栈、232. 用栈实现队列
【Leetcode】225. 用队列实现栈、232. 用栈实现队列
93 0
【Leetcode】225. 用队列实现栈、232. 用栈实现队列
|
存储 前端开发 算法
一行代码解决LeetCode实现 strStr()使用JavaScript解题|前端学算法
一行代码解决LeetCode实现 strStr()使用JavaScript解题|前端学算法
139 0
一行代码解决LeetCode实现 strStr()使用JavaScript解题|前端学算法
代码随想录刷题|LeetCode 栈和队列的理论基础 232.用栈实现队列 225. 用队列实现栈
代码随想录刷题|LeetCode 栈和队列的理论基础 232.用栈实现队列 225. 用队列实现栈
代码随想录刷题|LeetCode 栈和队列的理论基础 232.用栈实现队列 225. 用队列实现栈
|
存储 算法 索引
代码随想录刷题|LeetCode KMP算法理论 28. 实现 strStr() 459.重复的子字符串
代码随想录刷题|LeetCode KMP算法理论 28. 实现 strStr() 459.重复的子字符串
代码随想录刷题|LeetCode KMP算法理论 28. 实现 strStr() 459.重复的子字符串
leetcode 用栈实现队列
请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push、pop、peek、empty): 实现 MyQueue 类: void push(int x) 将元素 x 推到队列的末尾 int pop() 从队列的开头移除并返回元素 int peek() 返回队列开头的元素 boolean empty() 如果队列为空,返回 true ;否则,返回 false
52 0
LeetCode每日一题——676. 实现一个魔法字典
设计一个使用单词列表进行初始化的数据结构,单词列表中的单词 互不相同 。 如果给出一个单词,请判定能否只将这个单词中一个字母换成另一个字母,使得所形成的新单词存在于你构建的字典中。
62 0
【LeetCode】用 Rand7() 实现 Rand10()
【LeetCode】用 Rand7() 实现 Rand10()
108 0
leetcode【栈与队列—简单】 225. 用队列实现栈
leetcode【栈与队列—简单】 225. 用队列实现栈
leetcode【栈与队列—简单】 225. 用队列实现栈
leetcode【栈与队列—简单】 232. 栈实现队列
leetcode【栈与队列—简单】 232. 栈实现队列
leetcode【栈与队列—简单】 232. 栈实现队列