这篇文章主要讲解了“C++回溯算法中组合的相关问题怎么解决”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++回溯算法中组合的相关问题怎么解决”吧!
回溯算法模板
void backtracking(参数) {
if (终止条件) {
存放结果;
return;
}
for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {
处理节点;
backtracking(路径,选择列表); // 递归
回溯,撤销处理结果
}
}
回溯问题,最关键的是画出二叉树,遍历、剪枝问题都要通过直观的观察才能总结
剪枝策略
已经选择的元素个数:path.size();
还需要的元素个数为: k - path.size();
在集合n中至多要从该起始位置 : n - (k - path.size()) + 1,开始遍历
class Solution {
private:
vector<vector<int>> result;
vector<int> path;
void backtracking(int n,int k,int startIndex){
if(path.size()==k){
result.push_back(path);
return;
}
for(int i=startIndex;i<=n-(k-path.size())+1;i++){
path.push_back(i);
backtracking(n,k,i+1);
path.pop_back();
}
}
public:
vector<vector<int>> combine(int n, int k) {
backtracking(n,k,1);
return result;
}
};
在组合的基础上,多了一个求和的操作,求和也可以剪枝
class Solution {
private:
vector<vector<int>> result;
vector<int> path;
void backtracking(int sum,int k,int n,int startIndex){
if(sum>n) return;
if(path.size()==k){
if(sum==n) result.push_back(path);
return;
}
for(int i=startIndex;i<=9-(k-path.size())+1;i++){
path.push_back(i);
sum+=i;
backtracking(sum,k,n,i+1);
sum-=i;
path.pop_back();
}
}
public:
vector<vector<int>> combinationSum3(int k, int n) {
backtracking(0,k,n,1);
return result;
}
};
本题与组合III的区别在于,不限制组合内数字的个数,且同一个数字可以无限制重复被选取,体现在代码上就是,向下递归的时候,i不变
class Solution {
private:
vector<int> path;
vector<vector<int>> result;
void backtracking(vector<int>& candidates, int target,int index,int sum){
if(sum>target) return;
if(sum==target){
result.push_back(path);
return;
}
for(int i=index;i<candidates.size();i++){
path.push_back(candidates[i]);
sum+=candidates[i];
backtracking(candidates,target,i,sum);
sum-=candidates[i];
path.pop_back();
}
}
public:
vector<vector<int>> combinationSum(vector<int>& candidates, int target) {
backtracking(candidates,target,0,0);
return result;
}
};
本题和组合总和的区别在于,输入样例中含有重复元素时,输出样例不能有重复元素
同一条枝干上,元素可以相同;而不同的枝干则不能重复
即:横向遍历不能重复、纵向遍历可以重复
class Solution {
private:
vector<int> path;
vector<vector<int>> result;
void backtracking(vector<int>& candidates, int target,int index,int sum){
if(sum>target) return;
if(sum==target){
result.push_back(path);
return;
}
for(int i=index;i<candidates.size();i++){
if(i>index&&candidates[i]==candidates[i-1])
continue;
path.push_back(candidates[i]);
sum+=candidates[i];
backtracking(candidates,target,i+1,sum);
sum-=candidates[i];
path.pop_back();
}
}
public:
vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {
sort(candidates.begin(),candidates.end());
backtracking(candidates,target,0,0);
return result;
}
};
这题很好的考察了:for循环横向遍历、递归纵向遍历的知识点
class Solution {
private:
const string letterMap[10]={
"",
"",
"abc",
"def",
"ghi",
"jkl",
"mno",
"pqrs",
"tuv",
"wxyz"
};
public:
string path;
vector<string> result;
void backtracking(string digits,int index){
if(index==digits.size()){
result.push_back(path);
return;
}
int digit=digits[index]-'0';
string letter=letterMap[digit];
for(int i=0;i<letter.size();i++){
path.push_back(letter[i]);
backtracking(digits,index+1);
path.pop_back();
}
}
vector<string> letterCombinations(string digits) {
if(digits.size()==0)
return result;
backtracking(digits,0);
return result;
}
};
感谢各位的阅读,以上就是“C++回溯算法中组合的相关问题怎么解决”的内容了,经过本文的学习后,相信大家对C++回溯算法中组合的相关问题怎么解决这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!
亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。
原文链接:https://blog.csdn.net/BAI20010904/article/details/129481296