77. 组合
暴力方法为使用k个for循环遍历全部数据,
通过递归来实现 多个for循环
//fpo
class Solution {public:void backtracking(int n,int k,int startIndex){if(path.size()==k){result.push_back(path);return;}//for循环控制横向的遍历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) {backtracking(n,k,1);return result;}vector<vector<int>> result;vector<int> path;};
剪枝优化
来举一个例子,n = 4,k = 4的话,那么第一层for循环的时候,从元素2开始的遍历都没有意义了。 在第二层for循环,从元素3开始的遍历都没有意义了。
要保证剩余的元素还能组成k个数字。不满足条件的都可以不进行递归
优化过程如下:
- 已经选择的元素个数:path.size();
- 还需要的元素个数为: k - path.size();
- 在集合n中至多要从该起始位置 : n - (k - path.size()) + 1,开始遍历
优化后的循环为:
for (int i = startIndex; i <= n - (k - path.size()) + 1; i++) // i为本次搜索的起始位置
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;}};
