/*46.全排列
给定一个没有重复数字的序列,返回其所有可能的全排列。回溯大法!!
*/
void dfs(vector<int>& nums, int curSize,vector<vector<int>>& res, vector<int> &vec, map<int, int>& visited){
if(curSize == nums.size() ){
res.push_back(vec);
return;
}
for(int i = 0; i< nums.size(); i++){
if(!visited[i]){
vec.push_back(nums[i]);
visited[i] = 1;
dfs(nums, curSize+1,res ,vec, visited);
vec.pop_back();
visited[i] = 0;
}
}
}
vector<vector<int>> permute(vector<int>& nums) {
vector<vector<int>> res;
vector<int> vec;
map<int, int> visited;
dfs(nums, 0 , res ,vec, visited);
return res;
}
/*47 全排列 二
给定一个可包含重复数字的序列,返回所有不重复的全排列
输入: [1,1,2]
输出:
[
[1,1,2],
[1,2,1],
[2,1,1]
]
*/
class Solution {
public:
void dfs(int size, int curSize,vector<vector<int>>& res, vector<int> &vec, map<int, int>& visited){
if(curSize == size ){
res.push_back(vec);
return;
}
for (auto& p : visited) {
if (p.second == 0) continue;
--p.second;
vec.push_back(p.first);
dfs(size, curSize + 1 , res ,vec, visited);
++p.second;
vec.pop_back();
}
}
vector<vector<int>> permuteUnique(vector<int>& nums) {
vector<vector<int>> res;
vector<int> vec;
map<int, int> visited;
for (auto x : nums) ++visited[x];
dfs(nums.size(), 0 , res ,vec, visited);
return res;
}
};
/*39. 组合总和
给定一个无重复元素的数组?candidates?和一个目标数?target?,找出?candidates?中所有可以使数字和为?target?的组合。
candidates?中的数字可以无限制重复被选取。
所有数字(包括 target)都是正整数。
解集不能包含重复的组合。
输入: candidates = [2,3,6,7], target = 7,
所求解集为:
[
[7],
[2,2,3]
]
*/
/*class Solution {
public:
void dfs(vector<int>& candidates, int target,vector<vector<int>>& res, vector<int>& vec, int curSize){
if(target == curSize){
res.push_back(vec);
}
for(int i= 0; i< candidates.size(); i++){
if(curSize > target)
break;
vec.push_back(candidates[i]);
dfs(candidates, target, res, vec, curSize + candidates[i]);
vec.pop_back();
}
}
vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
vector<vector<int>> res;
vector<int> vec;
sort(candidates.begin(),candidates.end());
dfs(candidates, target, res, vec, 0);
return res;
}
};*/ 不对
class Solution {
public:
void dfs(vector<int>& candidates, int target,vector<vector<int>>& res, vector<int>& vec, int curSize){
if(target == 0){
res.push_back(vec);
return;
}
for(int i= curSize; i< candidates.size() && (target -candidates[i] >= 0); i++){
vec.push_back(candidates[i]);
dfs(candidates, target-candidates[i], res, vec, i);
vec.pop_back();
}
}
vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
vector<vector<int>> res;
vector<int> vec;
sort(candidates.begin(),candidates.end());
dfs(candidates, target, res, vec, 0);
return res;
}
}; //组合总和, 第一种方法并没有确认之前用过的数,后面不能用,所以第二种方法要把索引带进去!,保证用过的不再使用!!
/*40. 组合总和
给定一个数组?candidates?和一个目标数?target?,找出?candidates?中所有可以使数字和为?target?的组合。
candidates?中的每个数字在每个组合中只能使用一次
说明:
所有数字(包括目标数)都是正整数。
解集不能包含重复的组合。
输入: candidates =?[10,1,2,7,6,1,5], target =?8,
所求解集为:
[
[1, 7],
[1, 2, 5],
[2, 6],
[1, 1, 6]
]
*/
void dfs(vector<int>& candidates, int target,vector<vector<int>>& res, vector<int>& vec, int curSize){
if(target == 0){
res.push_back(vec);
return;
}
if (target < 0) return;
for(int i= curSize; i< candidates.size() && (target -candidates[i] >= 0); i++){
if (i != curSize && candidates[i] == candidates[i - 1]) //判断相邻元素是否相等,相等则contiue
continue;
vec.push_back(candidates[i]);
dfs(candidates, target-candidates[i], res, vec, i+1); //元素不可重复利用,那么i取i+1
vec.pop_back();
}
}
vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {
vector<vector<int>> res;
vector<int> vec;
sort(candidates.begin(),candidates.end());
dfs(candidates, target, res, vec, 0);
return res;
}
/*78.子集
给定一组不含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。
说明:解集不能包含重复的子集。
输入: nums = [1,2,3]
输出:
[
[3],
[1],
[2],
[1,2,3],
[1,3],
[2,3],
[1,2],
[]
]
*/
class Solution {
public:
vector<vector<int>> subsets(vector<int>& nums) {
vector<vector<int>> ans;
int len = nums.size(), last = 1 << len;
for(int i=0; i<last; ++i){
vector<int> curr;
for(int j=0; j<len; ++j){
if(i & (1 << len-j-1)){
curr.push_back(nums[j]);
}
}
ans.push_back(curr);
}
return ans;
}
};
class Solution {
public:
vector<vector<int>> ans; //用于存放最终结果
vector<int> path; // 深度优先搜索路径
vector<vector<int>> subsets(vector<int>& nums) {
sort(nums.begin(), nums.end());// 首先进行排序,方便后续处理
dfs(nums, 0, path); // 深度优先搜索过程
return ans;
}
void dfs(vector<int>& nums, int start, vector<int> tmp) {
ans.push_back(tmp); // 每递归一层,先把当前路径放到结果中(因为是要拿到所有的子集)
for (int i = start; i < nums.size() ; ++i) { // 对每一层进行遍历
// if (i > start && nums[i] == nums[i-1]) continue; // 这里的i > start是关键,一会讲
tmp.push_back(nums[i]);
dfs(nums, i + 1, tmp);
tmp.pop_back(); // 恢复原状
}
}
};
/*90 子集二
给定一个可能包含重复元素的整数数组 nums,返回该数组所有可能的子集(幂集)。
说明:解集不能包含重复的子集。
输入: [1,2,2]
输出:
[
[2],
[1],
[1,2,2],
[2,2],
[1,2],
[]
]
*/
class Solution {
public:
vector<vector<int>> ans; //用于存放最终结果
vector<int> path; // 深度优先搜索路径
vector<vector<int>> subsetsWithDup(vector<int>& nums) {
sort(nums.begin(), nums.end());// 首先进行排序,方便后续处理
dfs(nums, 0, path); // 深度优先搜索过程
return ans;
}
void dfs(vector<int>& nums, int start, vector<int> tmp) {
ans.push_back(tmp); // 每递归一层,先把当前路径放到结果中(因为是要拿到所有的子集)
for (int i = start; i < nums.size() ; ++i) { // 对每一层进行遍历
if (i > start && nums[i] == nums[i-1]) continue; // 这里的i > start是关键,一会讲
tmp.push_back(nums[i]);
dfs(nums, i + 1, tmp);
tmp.pop_back(); // 恢复原状
}
}
};