lc031: 下一个排列

题目描述

实现获取下一个排列的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列。如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列)。必须原地修改,只允许使用额外常数空间。

解法:

/**
 * 下一个排列总是会比上一个排列大,除非该排列已经是最大的排列
 * 我们找到一个比当前排列大的一个新序列,且变化的幅度尽可能小
 * 1:将左边的一个较小数与右边的一个较大数交换,让当前排列变大
 * 2:同时要让这个较小数尽量靠右,较大数尽可能小
 * 3:交换完成后,这个较大数右边的数需要重新按照升序排列
 */
// @lc code=start
class Solution {
    public void nextPermutation(int[] nums) {
        int i = nums.length - 2;
        //找到较小数的位置i
        while (i >= 0 && nums[i] >= nums[i + 1]) {
            i--;
        }
        //找到较大数的位置j
        if(i >= 0) {
            int j = nums.length - 1;
            while (j >= 0 && nums[i] >= nums[j]) {
                j--;
            }
            swap(nums, i, j);
        }
        reverse(nums, i + 1);
    }

    public void swap(int[] nums, int i, int j) {
        int temp = nums[i];
        nums[i] = nums[j];
        nums[j] = temp;
    }

    public void reverse(int[] nums, int start) {
        int left = start;
        int right = nums.length - 1;
        while (left < right) {
            swap(nums, left, right);
            left++;
            right--;
        }
    }
}

  

posted @ 2021-08-09 21:41  布尔先生  阅读(39)  评论(0编辑  收藏  举报