Find Permutations using Backtracking
This is a classic set of problems that can be solved using Backtracking.
Type 1 Problem: Permutation of Distinct Elements
Problem Description
Given a collection of distinct integers, return all possible permutations.
Input
[1,3,5]
Output
[
[1,3,5],
[1,5,3],
[3,1,5],
[3,5,1],
[5,1,3],
[5,3,1]
]
Solution
class Solution {
List<List<Integer>> results;
public List<List<Integer>> permute(int[] nums) {
this.results = new ArrayList<>();
// use a count array nad uniquenums array
int[] count = new int[nums.length];
Arrays.fill(count, 1);
// create a dummy result so we know the length of the ArrayList and we can use set() instead
// would be nice if java supports new ArrayList<>(Arrays.asList(nums)), but its a no-no due to boxing issue
List<Integer> result = new ArrayList<>();
for(int num: nums) result.add(num);
// call backtrack
backtrack(result, nums, count, 0);
return this.results;
}
private void backtrack(List<Integer> result, int[] nums, int[] count, int level){
if(level == nums.length){
results.add(new ArrayList(result));
return;
}
for(int i = 0; i< nums.length; i++){
if(count[i] == 0) continue; // the digit has been used before
result.set(level, nums[i]);
count[i]--;
backtrack(result, nums, count, level+1);
count[i]++;
}
}
}
Type 2 Problem: Permutation of Elements (duplicates possible)
Problem Description
Given a collection of integers that might contain duplicates, return all possible permutations.
Input
[1,1,5]
Output
[
[1,1,5],
[1,5,1],
[5,1,1]
]
Solution
class Solution {
List<List<Integer>> results;
public List<List<Integer>> permuteUnique(int[] nums) {
this.results = new ArrayList<>();
// handle duplicates
HashMap<Integer, Integer> hm = new HashMap<>();
for(int num : nums){
hm.put(num, hm.getOrDefault(num, 0)+1);
}
// use a count array nad uniquenums array
int[] uniqueNums = new int[hm.size()];
int[] count = new int[hm.size()];
int i = 0;
for(Map.Entry<Integer,Integer> entry : hm.entrySet()){
uniqueNums[i] = entry.getKey();
count[i] = entry.getValue();
i++;
}
// create a dummy result
List<Integer> result = new ArrayList<>();
for(int num: nums) result.add(num);
// call backtrack
backtrack(result, uniqueNums, count, 0);
return this.results;
}
private void backtrack(List<Integer> result, int[] nums, int[] count, int level){
if(level == result.size()){
results.add(new ArrayList(result));
return;
}
for(int i = 0; i< nums.length; i++){
if(count[i] == 0) continue; // the digit has been used before
result.set(level, nums[i]);
count[i]--;
backtrack(result, nums, count, level+1);
count[i]++;
}
}
}