47.全排列II
主要需要解决全排列不重复的问题,设定一个规则,保证在填第i个数的时候重复数字只会被填入一次即可,而在本题中,我们选择对原数组排序,保证相同的数字都相邻,然后每次填入的数一定是这个数所在重复数集合中「从左往右第一个未被填过的数字」
class Solution {
public List<List<Integer>> permuteUnique(int[] nums) {
List<List<Integer>> res = new ArrayList<List<Integer>>();
List<Integer> tmp = new ArrayList<Integer>();
int n = nums.length;
boolean[] visites = new boolean[n];
Arrays.sort(nums);
backtrack(res,tmp,visites,nums);
return res;
}
public void backtrack(List<List<Integer>> res,List<Integer> tmp,boolean[] visites,int[] nums){
if(tmp.size() == nums.length){
res.add(new ArrayList(tmp));
return;
}
for(int i = 0; i < nums.length; i++){
//由于会有重复,保证在填第i个数的时候重复数组只会被填入一次即可
//选择对原数字排序,保证相同的数字都相邻,然后每次填入的数一定是这个数所在重复数集合中从左往后第一个未被填过的数字
//!visites[i-1]限制两个相邻的重复数字的访问顺序
//比如[1,1,2]保证左边的1永远比右边的1先使用
/* vis[i]:当前数字是否出现过
* 如果当前数字与前一个数字相同(nums[i] == nums[i - 1]),
* 并且前一个数字还没有出现的话(vis[i - 1] == false),
* 那么就不能选择当前数字(continue),
* 如果前面的数字已经出现过(vis[i] == true),则可以
* 选择当前数字
*/
if(visites[i] || ( i>0 && nums[i] == nums[i -1] && !visites[i-1]))
{
continue;
}
tmp.add(nums[i]);
visites[i] = true;
backtrack(res,tmp,visites,nums);
tmp.remove(tmp.size()-1);
visites[i]=false;
}
}
}