「这是我参与11月更文挑战的第 22 天,活动详情查看:2021最后一次更文挑战」。
题目描述
这是 LeetCode 上的 384. 打乱数组 ,难度为 中等。
Tag : 「洗牌算法」
给你一个整数数组 nums
,设计算法来打乱一个没有重复元素的数组。
实现 Solution
class:
Solution(int[] nums)
使用整数数组nums
初始化对象int[] reset()
重设数组到它的初始状态并返回int[] shuffle()
返回数组随机打乱后的结果
示例:
1 | css复制代码输入 |
提示:
- 1<=nums.length<=2001 <= nums.length <= 2001<=nums.length<=200
- −106<=nums[i]<=106-10^6 <= nums[i] <= 10^6−106<=nums[i]<=106
nums
中的所有元素都是 唯一的- 最多可以调用 5∗1045 * 10^45∗104 次
reset
和shuffle
洗牌算法
共有 nnn 个不同的数,根据每个位置能够选择什么数,共有 n!n!n! 种组合。
题目要求每次调用 shuffle
时等概率返回某个方案,或者说每个元素都够等概率出现在每个位置中。
我们可以使用 KnuthKnuthKnuth 洗牌算法,在 O(n)O(n)O(n) 复杂度内等概率返回某个方案。
具体的,我们从前往后尝试填充 [0,n−1][0, n - 1][0,n−1] 该填入什么数时,通过随机当前下标与(剩余的)哪个下标进行值交换来实现。
对于下标 xxx 而言,我们从 [x,n−1][x, n - 1][x,n−1] 中随机出一个位置与 xxx 进行值交换,当所有位置都进行这样的处理后,我们便得到了一个公平的洗牌方案。
对于下标为 000 位置,从 [0,n−1][0, n - 1][0,n−1] 随机一个位置进行交换,共有 nnn 种选择;下标为 111 的位置,从 [1,n−1][1, n - 1][1,n−1] 随机一个位置进行交换,共有 n−1n - 1n−1 种选择 … 且每个位置的随机位置交换过程相互独立。
代码:
1 | Java复制代码class Solution { |
- 时间复杂度:O(n)O(n)O(n)
- 空间复杂度:O(n)O(n)O(n)
最后
这是我们「刷穿 LeetCode」系列文章的第 No.384
篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。
在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。
为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:github.com/SharingSour… 。
在仓库地址里,你可以看到系列文章的题解链接、系列文章的相应代码、LeetCode 原题链接和其他优选题解。
本文转载自: 掘金