目录
一、216. 组合总和 III
1.题目描述
找出所有相加之和为 n
的 k
个数的组合,且满足下列条件:
- 只使用数字1到9
- 每个数字 最多使用一次
返回 所有可能的有效组合的列表 。该列表不能包含相同的组合两次,组合可以以任何顺序返回。
示例 1:
输入: k = 3, n = 7 输出: [[1,2,4]] 解释: 1 + 2 + 4 = 7 没有其他符合的组合了。
示例 2:
输入: k = 3, n = 9 输出: [[1,2,6], [1,3,5], [2,3,4]] 解释: 1 + 2 + 6 = 9 1 + 3 + 5 = 9 2 + 3 + 4 = 9 没有其他符合的组合了。
示例 3:
输入: k = 4, n = 1 输出: [] 解释: 不存在有效的组合。 在[1,9]范围内使用4个不同的数字,我们可以得到的最小和是1+2+3+4 = 10,因为10 > 1,没有有效的组合。
2.解题思路
1.和力扣77.组合(这题代码我放在本篇文章第三题)那题很接近,只不过这里的终止条件要加一个对path中元素和的判断。
2.主要还是剪枝操作(不剪枝的话LeetCode判定超出时间限制),在77那题的基础上,多剪一个枝(如果当前path中元素和 已经大于题目要求的和,直接return)
3.代码实现
class Solution {
public:
vector<vector<int>> result;//结果集
vector<int> path;//存放符合条件的结果
void backtracking(int k,int n,int startIndex,int sum){
//剪枝操作
if(sum > n)
return;
//终止条件
if(path.size() == k && 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(k,n,i + 1,sum);
//回溯操作
path.pop_back();
sum -=i;
}
}
vector<vector<int>> combinationSum3(int k, int n) {
backtracking(k,n,1,0);
return result;
}
};
二、17. 电话号码的字母组合
1.题目描述
给定一个仅包含数字 2-9
的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。
给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。
示例 1:
输入:digits = "23" 输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]
示例 2:
输入:digits = "" 输出:[]
示例 3:
输入:digits = "2" 输出:["a","b","c"]
2.解题思路
1.先写一个二维数组,完成数字对字母的一一映射。2对应“abc”,3对应“def”......
2.终止条件:题目传来的电话号码中每个数字都一一映射过
3.单层逻辑:将index下标的字符转换成对应数字,然后映射到相应的字符串,开始for循环回溯递归。
3.代码实现
class Solution {
private:
const string letterMap[10] ={//数字和字符串的一一映射
"",//0
"",//1
"abc",//2
"def",//3
"ghi",//4
"jkl",//5
"mno",//6
"pqrs",//7
"tuv",//8
"wxyz",//9
};
public:
vector<string> result;
string s;
void backtracking(string digits,int index){
//终止条件:每个数字都做出了映射
if(index == digits.size()){
result.push_back(s);
return;
}
//单层逻辑
//找到下标为index的数字所对应的字符串,并将其存放到letter中
int digit = digits[index] - '0';
string letter = letterMap[digit];
for(int i = 0;i < letter.size();i++){
s.push_back(letter[i]);
backtracking(digits,index + 1);
s.pop_back();
}
}
vector<string> letterCombinations(string digits) {
if(digits.size() ==0)
return result;
backtracking(digits,0);
return result;
}
};
ps:不要忘记对digits做非空判断,否则会空指针异常。
三、77.组合
1.题目描述
1.题目描述
给定两个整数 n 和 k,返回范围 [1, n] 中所有可能的 k 个数的组合。
你可以按 任何顺序 返回答案。
示例 1:
输入:n = 4, k = 2
输出:
[
[2,4],
[3,4],
[2,3],
[1,2],
[1,3],
[1,4],
]
示例 2:
输入:n = 1, k = 1
输出:[[1]]
2.代码实现
class Solution {
public:
vector<vector<int>> result;//结果集
vector<int> path;//存放符合条件的结果
void backtracking(int n,int k,int startIndex){
//终止条件:一维数组中存放了k个元素
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) {
int startIndex = 1;
backtracking(n,k,startIndex);
return result;
}
};