回溯法:回溯搜索法,它是一种搜索的方式。回溯是递归的副产品,只要有递归就会有回溯。
回溯法解决的问题都可以抽象为树形结构,因为回溯法解决的都是在集合中递归查找子集,集合的大小就构成了树的宽度,递归的深度,都构成的树的深度。
回溯算法模板框架:
void backtracking(参数) {//回溯函数返回值一般是void,参数不同
if (终止条件) {//终止条件
存放结果;
return;
}
for (选择:本层集合中元素(树中节点孩子的数量就是集合的大小)) {//回溯遍历过程
处理节点;
backtracking(路径,选择列表); // 递归
回溯,撤销处理结果
}
}
思路:
1.递归函数的返回值以及参数:定义两个全局变量,一个用来存放符合条件单一结果path数组,一个用来存放符合条件结果的集合result数组。集合n里面取k个数,那么n和k是两个int型的参数,还需要一个参数,为int型变量startIndex,这个参数用来记录本层递归的中,集合从哪里开始遍历(集合就是[1,...,n] )。
2.回溯函数终止条件:path这个数组的大小如果达到k,说明我们找到了一个子集大小为k的组合。
3.单层搜索的过程:,for循环每次从startIndex开始遍历,然后用path保存取到的节点i,再进行递归处理,最后backtracking的下面部分就是回溯的操作了,撤销本次处理的结果。
代码:
class Solution {
public:
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; i++){
path.push_back(i);//处理节点
backtracking(n, k, i + 1);//递归
path.pop_back();//回溯,进行撤销操作,撤销处理的节点
}
}
vector<vector<int>> combine(int n, int k) {
result.clear();
path.clear();
backtracking(n, k, 1);
return result;
}
};