代码随想录day27 回溯继续深造,新题型

39. 组合总和

题目

给定一个无重复元素的数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的数字可以无限制重复被选取。

说明:

  • 所有数字(包括 target)都是正整数。
  • 解集不能包含重复的组合。

示例 1:

  • 输入:candidates = 2,3,6,7, target = 7,
  • 所求解集为: \[7, 2,2,3

思考

这题其实和组合总和III非常像,唯一的区别是数字可以无限制重复被选取,那么startIndex就不会作为去重的参数,每次递归都从i开始即可,其余的代码与组合总和III一模一样

代码

class Solution {

public:

vector<vector<int>> res;

vector<int> path;

void backTracking(vector<int>& candidates, int target, int sum, int startIndex) {

if(sum > target) return;

if(sum == target) {

res.push_back(path);

return;

}

for(int i = startIndex; i < candidates.size(); i++) {

sum += candidatesi;

path.push_back(candidatesi);

backTracking(candidates, target, sum, i);

sum -= candidatesi;

path.pop_back();

}

}

vector<vector<int>> combinationSum(vector<int>& candidates, int target) {

backTracking(candidates, target, 0, 0);

return res;

}

};

40.组合总和II

题目

给定一个数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合。

candidates 中的每个数字在每个组合中只能使用一次。

说明: 所有数字(包括目标数)都是正整数。解集不能包含重复的组合。

  • 示例 1:

  • 输入: candidates = 10,1,2,7,6,1,5, target = 8,

  • 所求解集为:

    [
    [1, 7],
    [1, 2, 5],
    [2, 6],
    [1, 1, 6]
    ]

思考

有难度,感觉和组合总和I、III不太一样,按照卡哥的话来说,这题要树层去重,即如果横向遍历时,如果该数与上一个数重复(注意需要把原数组sort一下),那么这个数可以考虑直接跳过,同时呢,为了树枝不去重(该数与上一个数相同也可以使用),我们这里用到了used数组的概念,这个used数组非常奇妙,其实相当于一个flag数组,大小和原数组一样,刚开始全是零,即所有数都没有使用过,当遍历到末尾时,used数组里全变成1,那么这时候要进入到下一个树层进行递归,又把原先遍历过的数置为零,这样当如果该数与上一个数重复且usedi-1 == 0时,才能真正跳过,这个去重的思路非常奇妙,也很典型,需要重点掌握

代码

class Solution {

public:

vector<vector<int>> res;

vector<int> path;

void backTracking(vector<int>& candidates, vector<int>& used, int target, int sum, int startIndex) {

if(sum > target) return;

if(sum == target) {

res.push_back(path);

return;

}

for(int i = startIndex; i < candidates.size(); i++) {

//下一行的去重逻辑是树层去重,usedi-1 == 0是要保证纵向遍历时树枝不去重

if(i > 0 && candidatesi == candidatesi-1 && usedi-1 == 0) continue;

sum += candidatesi;

path.push_back(candidatesi);

usedi = 1;

backTracking(candidates, used, target, sum, i+1);

sum -= candidatesi;

usedi = 0;

path.pop_back();

}

}

vector<vector<int>> combinationSum2(vector<int>& candidates, int target) {

vector<int> used(candidates.size(),0);

sort(candidates.begin(), candidates.end());//必须sort

backTracking(candidates, used, target, 0, 0);

return res;

}

};

131.分割回文串

题目

给定一个字符串 s,将 s 分割成一些子串,使每个子串都是回文串。

返回 s 所有可能的分割方案。

示例: 输入: "aab" 输出: \["aa","b", "a","a","b" ]

思考

和组合不一样,这题是分割,对于我来说难点有以下几个:

1、中止条件是什么? 因为要遍历到字符串最后一位,那么就是当startIndex == s.size()时收获借过

2、怎么收集到的字符串是否是回文?用双指针的方法来判断,及从要收集字符串头尾开始向中间遍历,如果si一直等于sj,那么就是回文串

3、判断是回文串时如何收集?用substr(startIndex, i - startIndex + 1)

代码

class Solution {

public:

vector<vector<string>> res;

vector<string> path;

bool isRound(const string& s, int start, int end) {//判断是否为回文串,双指针从头尾各自向中间移动

for(int i = start, j = end; i < j; i++, j--) {

if(si != sj) return false;

}

return true;

}

void backTracking(string s, int startIndex) {

if(startIndex == s.size()) {//当遍历到s末尾时开始收集结果

res.push_back(path);

return;

}

for(int i = startIndex; i < s.size(); i++) {

if(isRound(s, startIndex, i)) {

string tmp = s.substr(startIndex, i - startIndex + 1);//截取从stratIndex到i的字符串

path.push_back(tmp);

}

else continue;

backTracking(s, i+1);

path.pop_back();

}

}

vector<vector<string>> partition(string s) {

backTracking(s, 0);

return res;

}

};

相关推荐
QiLinkOS4 分钟前
发明人与专利价值共生逻辑
c语言·数据结构·c++·人工智能·单片机·嵌入式硬件·算法
计算机安禾15 分钟前
【算法分析与设计】第21篇:回溯法的状态空间树与剪枝函数设计
大数据·人工智能·算法·机器学习·数据挖掘·剪枝
磊 子17 分钟前
STL之set以及set和map区别
开发语言·c++·算法
Promise微笑26 分钟前
算法突围:“双核四驱”理论下的“官网”AI引用概率提升指南
人工智能·算法·chatgpt
KaMeidebaby32 分钟前
卡梅德生物技术快报|免疫共沉淀 - Co-IP 实验在转录因子 ATF3/Smad4 蛋白互作研究中的应用实例解析
网络·人工智能·网络协议·tcp/ip·其他·算法·新浪微博
wayz1137 分钟前
20260530 软件ETF(159852)量化分析
算法·金融·数据分析·量化交易
通信小呆呆1 小时前
Hankel结构及其快速算法详解
线性代数·算法·机器学习
四代水门1 小时前
服务端倒带(Server-Side Rewind)命中判定系统
java·前端·算法
吃好睡好便好1 小时前
矩阵的左除和右除
人工智能·学习·线性代数·算法·矩阵
江屿风1 小时前
C++图的基本概念流食般投喂-竞赛编
开发语言·数据结构·c++·笔记·算法·图论