leecode/daily/2019-07-19.md

36 lines
1.6 KiB
Markdown
Raw Normal View History

2020-05-22 18:17:19 +08:00
# 毎日一题 - 洗牌算法
## 信息卡片
* 时间2019-07-19
* 题目链接:暂无
* tag`Array` `Probability`
## 题目描述
```
假设我们有一个n个元素的数组要求你实现一个函数该函数会随机地返回n个元素的排列要求所有排列出现的概率是一样的。即每一个排列出现的概率都是1/n!.
```
## 参考答案
思路如下
像洗牌一样,从数组中随机取出一个,放入另一个全新的数组中,但这会涉及到数组删除操作.
在这个基础上转换一下思路把从数组中取出的元素放入原数组中第一次随机删除时把它与原数组的倒数第一个交换第2次在剩下的元素中随机删除时把它与原数组的倒数第2个交换第n-1次最后一次不用换时便完成了洗牌 时间复杂度为O(n)
```js
function shuffle(list) {
for (let i = list.length - 1; i >= 1; i--) {
const random = (Math.random() * (i + 1)) >> 0;
const temp = list[i];
list[i] = list[random];
list[random] = temp;
}
}
```
注: 概率证明, 任意一个元素放在倒数第一个位置的概率为1/n,放到倒数第2个的概率为 [(n-1)/n ]* [1/(n-1)] = 1/n,放在倒数第k个位置的概率是[(n-1)/n] * [(n-2)/(n-1)] *...* [(n-k+1)/(n-k+2)] *[1/(n-k+1)] = 1/n 因此每一个元素放在任意位置的概率都为1/n,所有的排列出现的概率则为 1/n * 1/(n-1) *..* 1 = 1/n!
在交换时之所以第一次与第n个交换不与第一个交换是因为与第n个交换代码更简洁
## 优秀解答
>暂缺