理论基础
77. 组合
class Solution {
public:
vector<vector<int>> ans;
vector<int> path;
void backtracking(int n, int k, int startindex){
if(path.size() == k){
ans.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();
}
}
vector<vector<int>> combine(int n, int k) {
backtracking(n, k, 1);
return ans;
}
};
剪枝操作
for(int i=startindex; i<=(n-(k-path.size())+1); i++)
可以剪枝的地方就在递归中每一层的for循环所选择的起始位置。
如果for循环选择的起始位置之后的元素个数 已经不足 我们需要的元素个数了,那么就没有必要搜索了。
-
已经选择的元素个数:path.size();
-
还需要的元素个数为: k - path.size();
-
在集合n中至多要从该起始位置 : n - (k - path.size()) + 1,开始遍历
为什么有个+1呢,因为包括起始位置,我们要是一个左闭的集合。
举个例子,n = 4,k = 3, 目前已经选取的元素为0(path.size为0),n - (k - 0) + 1 即 4 - ( 3 - 0) + 1 = 2。