给出一个数组,从新排列,返回『下一个排列,题目的描述中还给出了几个例子。数组
有一首歌名是下一个天亮,不过和这道题没什么关系。还有一类题是已有一堆数字要返回全部的排列方式,和这道题也没什么关系。按照题目的描述中的例子,好比:[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)。leetcode