实现获取下一个排列的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列。
如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列)。
必须原地修改,只允许使用额外常数空间。
以下是一些例子,输入位于左侧列,其相应输出位于右侧列。1,2,3
→ 1,3,2
3,2,1
→ 1,2,3
1,1,5
→ 1,5,1
思路:
有如下的一个数组
1 5 8 4 7 6 5 3 1
下一个排列为:
1 5 8 5 1 3 4 6 7
那么是如何得到的呢,我们通过观察原数组可以发现,如果从末尾往前看,数字逐渐变大,到了4时才减小的,然后我们再从后往前找第一个比4大的数字,是5,那么我们交换4和5,再把此时4后面的所有数字转置一下即可,步骤如下:
1 5 8 4 7 6 5 3 1
1 5 8 4 7 6 5 3 1
1 5 8 5 7 6 4 3 1
1 5 8 5 1 3 4 6 7
public void nextPermutation(int[] nums) {
if(nums == null || nums.length == 0) return;
int i = nums.length - 2;
while(i >= 0 && nums[i + 1] <= nums[i]){ //从右向左找到第一个开始减小的数字
i--;
}
if(i >= 0){
int j = nums.length - 1;
while(j >= 0 && nums[j] <= nums[i]) { //从右向左找到第一个比num[i]小的数
j--;
}
swap(nums, i, j); //交换
}
reverse(nums, i + 1); //将i位置后的数字转置
}
public static void reverse(int[] nums, int start){
int i = start, int j = nums.length - 1;
while(i < j){
swap(nums, i, j);
i++;
j--;
}
}
public static void swap(int[] nums, int i, int j){
int temp = nums[i];
nums[i] = nums[j];
nums[j] = temp;
}