leetcode 刷题

/*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(); // 恢复原状
        }
    }
};
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值