Next Permutation LC解题记录

题目内容

给出一个数组,从新排列,返回『下一个排列,题目的描述中还给出了几个例子。数组

解决思路

有一首歌名是下一个天亮,不过和这道题没什么关系。还有一类题是已有一堆数字要返回全部的排列方式,和这道题也没什么关系。按照题目的描述中的例子,好比:
[1,2,3] -> [1,3,2] 观察了一下,是交换了2和3的位置。
[3,2,1] -> [1,2,3]这个就是整个数组的逆序排列。
根据这两个例子猜想,须要两个辅助的方法,一个是交换(swap),另外一个是逆序(reverse)。
继续看,交换的地方应该是相邻两个数字后一个比前一个大的状况,并且靠后的优先,好比[#$%,1,2,3]返回的也是[#$%,1,3,2]。因此第一步的思路就是从后往前找,找一对儿符合要求的相邻数字。若是找到头都没找到,那能够直接逆序输出返回结果了。
找到这对儿以后,小的数字确定要交换到后面的,先设定这个小数字的位置是first。而后要交换的数字是first后面比它大的里面最小的,就是贴着头皮理发的感受。交换过来以后,再把first后面的部分逆序输出就行了。
这道题的关键在于,找到规律,数学上的规律。关键的关键在于,找不到规律就看discussion吧。code

code

public class Solution {
    public void nextPermutation(int[] nums) {
        //排除corner case
        if(nums == null || nums.length < 2) return;
        //找那对儿数字
        int right = nums.length-1;
        while(right > 0){
            if(nums[right-1] < nums[right]){
                break;
            }
            right--;
        }
        //没找着符合要求的状况,直接reverse。
        if(right == 0){
            reverse(nums, 0, nums.length-1);
            return;
        }
        
        int first = right-1;
        int nextBig = right;
        while(right < nums.length){
            //要大,还不能太大。
            if(nums[right] <= nums[nextBig] && nums[right] > nums[first]){
                nextBig = right;
            }
            right++;
        }
        swap(nums, first, nextBig);
        reverse(nums, first+1, nums.length-1);
    }
    
    private void swap(int[] nums, int first, int second){
        int temp = nums[first];
        nums[first] = nums[second];
        nums[second] = temp;
        return;
    }
    private void reverse(int[] nums,int start, int end){
        while(start < end){
            int temp = nums[start];
            nums[start] = nums[end];
            nums[end] = temp;
            start++;
            end--;
        }
    }
}

复杂度分析

这个很显然是O(n)了,第一遍找一对儿数字的时候扫了一遍,而后第二次再找nextBig的时候又扫了一遍,最后逆序输出,扫了好几回,最后依然是O(n)。leetcode

相关文章
相关标签/搜索