Next Permutation LC解题记录

377 查看

题目内容

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

解决思路

有一首歌名是下一个天亮,不过和这道题没什么关系。还有一类题是已有一堆数字要返回所有的排列方式,和这道题也没什么关系。按照题目的描述中的例子,比如:
[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

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)。