今天我们继续刷Leetcode 热题 HOT 100,日复一日,相信自己,一定会有进步。如果一个人刷题太孤独了,欢迎加群每日一题算法群,让我们大家一起监督,一起成长。
Leetcode - 31.下一个排列
题目描述:
实现获取下一个排列的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列。
如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列)。
必须原地修改,只允许使用额外常数空间。
以下是一些例子,输入位于左侧列,其相应输出位于右侧列。
示例:
1,2,3
→ 1,3,2
3,2,1
→ 1,2,3
1,1,5
→ 1,5,1
解题思路:
怎么解呢?
方法一:暴力法
在这种方法中,我们找出由给定数组的元素形成的列表的每个可能的排列,并找出比给定的排列更大的排列。
但是这个方法是一种非常天真的方法,因为它要求我们找出所有可能的排列
因此,这种方法根本无法通过。 所以,我们直接采用正确的方法。
复杂度分析
时间复杂度:O(n!),可能的排列总计有 n! 个。
空间复杂度:O(n),因为数组将用于存储排列。
方法二:一遍扫描
首先,我们观察到对于任何给定序列的降序,没有可能的下一个更大的排列。
例如,以下数组不可能有下一个排列:
[9, 5, 4, 3, 1]
此题的目的是求一组元素可以组成的所有数字中比这组元素组成的数字下一大的一组序列
1.一种特殊情况:当序列的元素递减的时候肯定是不存在比它大的序列了,像[3,2,1]组成的数字321已经是最大的了
2.当不是上面的特殊情况的时候,举个例子:
[1,3,2,4]的下一大序列是[1,3,4,2]
[1,3,4,2]的下一大序列是[1,4,2,3]
[1,4,3,2]的下一大序列是[2,1,3,4]
所以我们要从上面找到规律
从上面,我们可以发现规律,从序列的后面向前面看,如果nums[i]>nums[i-1]那么这个序列就存在下一大元素
我们希望下一个数比当前数大,这样才满足“下一个排列”的定义。因此只需要将后面的大数与前面的小数交换,就能得到一个更大的数。比如 123456
,将 5
和 6
交换就能得到一个更大的数 123465
。
我们还希望下一个数增加的幅度尽可能的小,这样才满足“下一个排列与当前排列紧邻“的要求。为了满足这个要求,我们需要:
在尽可能靠右的低位进行交换,需要从后向前查找
将一个尽可能小的大数与前面的小数交换。比如 123465
,下一个排列应该把 5 和 4 交换而不是把 6
和 4
交换
将大数换到前面后,需要将大数后面的所有数重置为升序,升序排列就是最小的排列。以 123465
为例:首先按照上一步,交换 5
和 4
,得到 123564
;然后需要将 5
之后的数重置为升序,得到 123546
。显然 123546
比 123564
更小,123546
就是 123465
的下一个排列
标准的“下一个排列”算法可以描述为:
class Solution {
public void nextPermutation(int[] nums) {
if(nums.length==0)
return;
int firstindex=-1;
for(int i=nums.length-2;i>=0;i--){
if(nums[i]=0;i--){
if(nums[firstindex]
往期回顾
LeetCode day 1 题号1、2(两数之和,两数相加)
LeetCode day 2 题号 3、4 (最长无重复子串,两个有序数组的中位数)
LeetCode day3 题号5 (最长回文子串)
LeetCode day 4 10.正则匹配
LeetCode day 5 盛最多水的容器(双指针)
LeetCode day 6 三数之和=两数之和plus
LeetCode day7 初入搜索
LeetCode day 8 19、20
LeetCode day 9 合并两个有序链表
扫码加入我们