代码随想录刷题day27|组合总和II组合总和II分割回文串

文章目录

  • day27学习内容
  • 一、组合总和-所选数字可重复
    • 1.1、代码-正确写法
    • 1.1.1、为什么递归取的是i而不是i+1呢?
  • 二、组合总和II-所选数字不可重复
    • 2.1、和39题有什么不同
    • 2.2、思路
      • 2.2.1、初始化
      • 2.2.2、主要步骤
      • 2.2.3、回溯函数 `backTracking`
    • 2.3、正确写法1
      • 2.3.1、为什么要先排序
      • 2.3.2、去重为什么是!used[i - 1]而不是used[i - 1]
  • 三、分割回文串
    • 3.1、科普-什么是回文串?
    • 3.2、思路
    • 3.3、代码-正确写法
    • 3.3.1、为什么看到deque结果就往最终的result里面add?
    • 3.3.2、为什么判断回文串是从i开始,截取却从i+1开始呢?
  • 总结
    • 1.感想
    • 2.思维导图


day27学习内容

day27主要内容

  • 组合总和
  • 组合总和II
  • 分割回文串

声明
本文思路和文字,引用自《代码随想录》

一、组合总和-所选数字可重复

39.原题链接

1.1、代码-正确写法

class Solution {List<List<Integer>> result = new ArrayList();List<Integer> path = new ArrayList();public List<List<Integer>> combinationSum(int[] candidates, int target) {backTracking(candidates, target, 0, 0);return result;}private void backTracking(int[] candidates, int target, int sum, int startIndex) {// 累加和大于target,直接returnif (sum > target) {return;}if (sum == target) {result.add(new ArrayList(path));return;}for (int i = startIndex; i < candidates.length; i++) {path.add(candidates[i]);// 递归backTracking(candidates, target, sum + candidates[i], i);// 回溯path.removeLast();}}
}

1.1.1、为什么递归取的是i而不是i+1呢?

看题意,在这里插入图片描述
题目中说的很清楚了,可以无限制重复选取,所以不是从n+1开始迭代的。

二、组合总和II-所选数字不可重复

40.原题链接

2.1、和39题有什么不同

不同点1:看题目,原始数组中有重复的元素。
在这里插入图片描述
不同点2:累加求和的时候,可以用同样的元素。但是很重要的一点是每个数字在每个组合中只能使用一次。

2.2、思路

  • 对树层进行去重,对树枝不去重

2.2.1、初始化

  • LinkedList<Integer> path:用于临时存储当前探索路径中的数字序列。
  • List<List<Integer>> ans:用来存储所有符合条件的组合。
  • boolean[] used:标记数组中的元素是否被使用过,以避免在同一层中重复使用相同的元素。
  • int sum:记录当前路径上所有元素的总和。

2.2.2、主要步骤

  1. 排序:首先对数组 candidates 进行排序,这是为了实现有效去重和提前终止搜索。
  2. 调用回溯函数:通过 backTracking 函数开始回溯搜索。

2.2.3、回溯函数 backTracking

  • 结束条件:当 sum(当前路径元素的总和)等于 target 时,将当前路径复制一份并添加到结果集 ans 中,然后返回上一层继续探索。
  • 遍历候选数组:从 startIndex 开始遍历 candidates 数组,以避免重复使用元素。
  • 剪枝条件
    • 如果当前元素加上 sum 超过了 target,则由于数组已经排序,后面的元素只会更大,所以直接中断当前循环。
    • 如果当前元素与前一个元素相同,且前一个元素未被使用过,则跳过当前元素。这一步是去重的关键。
  • 递归与回溯
    • 首先标记当前元素为已使用,将其加入 path,并更新 sum
    • 然后递归调用 backTracking,注意 startIndex 更新为 i + 1,因为每个元素只能使用一次。
    • 递归返回后,撤销当前元素的选择:将其标记为未使用,从 path 中移除,并更新 sum

通过这样的递归+回溯的方式,加上有效的剪枝条件,代码能够高效地搜索整个解空间,找出所有满足条件的组合。排序是去重和剪枝的基础,确保了算法的高效和正确性。

2.3、正确写法1

class Solution {LinkedList<Integer> path = new LinkedList<>();List<List<Integer>> ans = new ArrayList<>();boolean[] used;int sum = 0;public List<List<Integer>> combinationSum2(int[] candidates, int target) {used = new boolean[candidates.length];// 加标志,用来判断同层节点是否已经遍历Arrays.fill(used, false);// 去重必须先排序Arrays.sort(candidates);backTracking(candidates, target, 0);return ans;}private void backTracking(int[] candidates, int target, int startIndex) {if (sum == target) {ans.add(new ArrayList(path));}for (int i = startIndex; i < candidates.length; i++) {if (sum + candidates[i] > target) {break;}// 去重逻辑if (i > 0 && candidates[i] == candidates[i - 1] && !used[i - 1]) {continue;}used[i] = true;sum += candidates[i];path.add(candidates[i]);// 每个节点仅能选择一次,所以从下一位开始backTracking(candidates, target, i + 1);used[i] = false;sum -= candidates[i];path.removeLast();}}
}

2.3.1、为什么要先排序

  • 去重逻辑的实现
    • 排序是实现去重逻辑的前提。当数组被排序后,重复的数字会被排列在一起。
    • 在回溯过程中,为了避免选择相同的数字造成的重复组合,需要检查当前数字是否与前一个数字相同,如果相同且前一个数字没有被使用(即在这一层中还未使用过),则跳过当前数字
    • 如果数组未排序,重复的数字可能散布在数组中的不同位置,我们就不能通过比较当前数字和前一个数字是否相同来有效地去重。
  • 提前终止搜索
    • 排序还允许我们在确定总和已经超过目标值时提前终止搜索。因为数组是有序的,
    • 如果在某一点上,当前路径上的数字总和加上当前考虑的数字已经超过了目标值,那么对于当前数字之后的所有数字,它们加入路径的结果也都会超过目标值(因为它们都是非负的,并且等于或大于当前数字)。
    • 这样可以避免无谓的搜索,从而提高算法的效率。

2.3.2、去重为什么是!used[i - 1]而不是used[i - 1]

看卡尔大佬的图,或者自己画个图,就很好理解了。简单来说,就是当你递归到最后一层往回回溯时,之前的节点,used的状态会重新变回[0]

三、分割回文串

131.原题链接

3.1、科普-什么是回文串?

来自gpt的解释:回文是一种特殊的字符串,它从前往后读和从后往前读是完全相同的。回文不仅限于单词和短语,也可以是一系列数字、字符或者其他序列,它们的特点是对称性。这种对称可以是字符级的,也就是说,忽略空格、标点符号和大小写后,字符串的前半部分和后半部分是镜像对称的。

3.2、思路

  • 什么时候分割结束
    • 看卡尔画出来的树图或者自己画一个,很明显到了字符串的末尾(可以理解为是叶子结点),就说明分割结束了。
  • 那么什么时候可以返回结果
    • 很明显,是回文字符串的时候,才能放入结果集。
  • 综上所述,这道问题就变成了判断每一个叶子结点是不是回文子串。如果叶子结点是回文字符串,就直接返回即可。
  • 怎么切割字符串
    • S0tring.subString()这方法应该知道吧。问题在于从哪里切割到哪里,
    • 答案是从startIndex切割到i。而且是左闭右闭。为什么呢?看3.3.2。

3.3、代码-正确写法

class Solution {List<List<String>> result= new ArrayList<>();Deque<String> deque = new LinkedList<>();public List<List<String>> partition(String s) {backTracking(s, 0);return result;}private void backTracking(String s, int startIndex) {// 如果起始位置大于s的大小,说明找到了一组分割方案if (startIndex >= s.length()) {//为什么看到结果就add了呢?请看3.3.1result.add(new ArrayList(deque));return;}for (int i = startIndex; i < s.length(); i++) {// 判断是不是回文串,如果是就加入到deque里面if (isPalindrome(s, startIndex, i)) {String str = s.substring(startIndex, i + 1);deque.addLast(str);} else {continue;}// 起始位置+1,不能取已经取过的字符串。backTracking(s, i + 1);deque.removeLast();}}// 判断是否是回文串// 用双指针法private boolean isPalindrome(String s, int startIndex, int end) {for (int i = startIndex, j = end; i < j; i++, j--) {if (s.charAt(i) != s.charAt(j)) {return false;}}return true;}
}

3.3.1、为什么看到deque结果就往最终的result里面add?

因为判断回文子串,是在单层逻辑里面判断的,在deque里面的字符串,一定是回文子串,不是回文子串不会往里面add的。所以可以无脑add。

3.3.2、为什么判断回文串是从i开始,截取却从i+1开始呢?

  • 在这段代码中,判断回文串是从startIndexi(包括i),这是因为需要检查从startIndexi的子字符串是否构成回文串。
  • 一旦确认这个子字符串是回文串,就加入deque中。
  • 截取从i+1开始是因为在回溯过程中,当我们确定了一个回文子字符串后,下一步应该从这个子字符串的下一个字符开始继续寻找新的回文子字符串。这样,我们可以确保不会重复使用已经被包含在找到的回文子字符串中的字符。
  • 例如,对于字符串"aab",当startIndex为0,且我们发现"aa"是回文时,下一次递归调用backTracking时的startIndex应该是2,即"aa"之后的字符,这就是为什么截取开始的索引是i+1。这样做可以避免重复处理相同的字符,确保每次递归都是针对剩余未处理的字符串部分。

总结

1.感想

  • 分割回文串这题好难,不咋会写,另外吐槽一下,卡尔老师的视频反光好严重。

2.思维导图

本文思路引用自代码随想录,感谢代码随想录作者。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/754198.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【 如何使用路由建立多视图单页应用】的详细讲解

使用路由建立多视图单页应用 1. 多视图1.1 引入依赖库1.2 创建自定义组件 2. React&#xff08;使用React Router&#xff09;3. Angular&#xff08;使用Angular Router&#xff09;4. Vue&#xff08;使用Vue Router&#xff09; 1. 多视图 构建多视图的单页应用程序&#x…

传值、传址、空间释放详细图解

目录 前言 一.进程 1.1 进程的映射 1.2 进程的虚拟空间 二.函数传参 2.1 函数传参 2.2 函数传值 2.2.1 函数传值案例1 2.2.2 函数传值案例2 2.2.3 返回值为常量 2.3 函数传送地址 2.3 字符串使用 前言 详细介绍函数传值和传地址区别&#xff1a;进行数据操作的区别&#xff0c…

c++算法学习笔记 (11) 离散化

1.前置知识&#xff1a; unique是 c标准模板库STL中十分实用的函数之一&#xff0c;使用此函数需要#include <algorithm>头文件 该函数的作用是“去除”容器或者数组中相邻元素的重复出现的元素。 (1) 这里的去除并非真正意义的erase&#xff0c;而是将重复的元素放到容…

VMware workstation pro 16 虚拟机的安装

VMware workstation pro 16 虚拟机的安装 VMware 16下载VMware 16安装VMware 16许可 VMware 16下载 下载地址&#xff1a; VMware workstation pro 16 官网下载地址 VMware 16安装 安装向导&#xff0c;点击下一步勾选我同意许可协议中的条款&#xff0c;点击下一步 更改安…

被奔驰看上的“人”

上周五&#xff0c;人形机器人公司、NASA 合作伙伴 Apptronik 宣布已与梅赛德斯奔驰&#xff08;以下简称奔驰&#xff09;达成一项商业协议&#xff0c;试点将身高 1.7 米、体重 140 多斤的双足机器人 Apollo 用于制造业。奔驰也成为继宝马、蔚来汽车之后最新尝试人形机器人的…

nodejs实现链接shp的属性信息(替换字段或者追加字段)

写在前面 偶尔听到同事说一个数据处理重复性的流程太多&#xff0c;就了解了下&#xff0c;原来是1份shp数据对应着一个xls属性文件&#xff0c;需要把xls文件里的一部分属性更新到shp的相关字段中&#xff0c;当然这种操作是可以用ArcGIS或者QGIS实现&#xff0c;但是当数据特…

综合交易模型----可转债三低策略实盘qmt版,提供源代码

链接 综合交易模型----可转债三低策略实盘qmt版&#xff0c;提供源代码 (qq.com) 可转债3低策略是指选择正股市值低、转债余额低、溢价率低的可转债进行投资的策略。 市值低&#xff1a;指的是可转债对应的正股市场价值较小&#xff0c;这通常意味着需要较少的资金就可以对股价…

jeecg启动Sentinel 一直是空白页面 解决办法用 外部 Sentinel SpringCloud之Sentinel概述和安装及简单整合

jeecg启动Sentinel 一直是空白页面 解决办法用 外部 Sentinel SpringCloud之Sentinel概述和安装及简单整合 文章目录 jeecg启动Sentinel 一直是空白页面 解决办法用 外部 Sentinel SpringCloud之Sentinel概述和安装及简单整合 Sentinel概述基本介绍 Sentinel安装下载地址: http…

聊聊AI时代学习这件事本身应该发生什么样的变化

随着 AI 大模型 的爆发&#xff0c;我们身处这个时代&#xff0c;应该怎么样去学习去了解这些前言的技术&#xff1f;可能很多人会说我英文不好&#xff0c;我算法不行&#xff0c;无法深入去了解 AI 大模型相关的知识吧&#xff1f; 没关系&#xff0c;其实博主也跟大家一样&…

注册登录Vue3+ElementPlus代码

1、app.vue <script setup>import LoginVue from /views/Login.vue </script><template><LoginVue/> </template> 2、Login.vue <script setup> import { User, Lock } from "element-plus/icons-vue"; import { ElMessage } …

[小程序开发] 外部样式类

一、外部样式类 在使用组件时&#xff0c;组件使用者可以给组件传入CSS类名&#xff0c;通过传入的类名修改组件的样式。 如果需要使用外部样式类修改组件的样式&#xff0c;在Component中需要用extemalClasses定义若干个外部样式类。 二、使用步骤 1、在Component中用externa…

C++面试100问!(一)

STL中MAP数据存放形式&#xff1f; 红黑树。unordered map底层结构是哈希表。 STL有什么基本组成&#xff1f; 容器、迭代器、仿函数、算法、分配器、配接器&#xff0c; 他们之间的关系&#xff1a;分配器给容器分配存储空间&#xff0c;算法通过迭代器获取容器中的内容&…

实在智能Agent——RPA终极进化方向

RPA技术备受瞩目&#xff0c;它通过“机器人”自动化了人力执行的重复性、低复杂度任务&#xff0c;解放了员工并降低了企业成本。RPA机器人全天候运行&#xff0c;避免人为错误&#xff0c;高效处理任务&#xff0c;成为处理事务、操作数据、回应查询的理想选择。在管理后台&a…

第四百一十回

文章目录 1. 概念介绍2. 方法与细节2.1 获取方法2.2 使用细节 3. 示例代码4. 内容总结 我们在上一章回中介绍了"如何获取当前系统语言"相关的内容&#xff0c;本章回中将介绍如何获取时间戳.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章…

实体店运营方案大全:从选址到日常管理的全面指导

想开实体店或正在创业的朋友们&#xff0c;大家好&#xff01;今天&#xff0c;我作为一个鲜奶吧5年的创业者&#xff0c;为大家分享一份实体店运营方案大全&#xff0c;涵盖从选址到日常管理的各个方面。我在实体店方面有一些心得体会&#xff0c;今天就来和大家聊聊&#xff…

【面试自测】Java并发

Synchronized 用 过 吗 &#xff0c; 其 原 理 是 什 么 &#xff1f;获 取 对 象 的 锁 &#xff0c; 这 个 “ 锁 ” 到 底 是 什 么 &#xff1f; 如 何 确 定 对 象 的 锁 &#xff1f;什 么 是 可 重 入 性 &#xff0c; 为 什 么 说 Synchronized 是 可 重 入 锁 &#x…

【软考】系统集成项目管理工程师(十九)收尾管理【1分】

一、项目收尾的内容 二、总结 三、软件后续工作 练一练 【例1-17上】&#xff08;B&#xff09;不属于项目验收的内容。 A.验收测试 B.系统维护工作 C.项目终验 D.系统试运行 【例2-17上】信息系统集成项目完成验收后要进行一个综合性的项目后评估,评估的内容一般包括&#…

uView LoadingIcon 加载动画

此组件为一个小动画&#xff0c;目前用在uView的loadMore加载更多等组件的正在加载状态场景。 #平台差异说明 App&#xff08;vue&#xff09;App&#xff08;nvue&#xff09;H5小程序√√√√ #基本使用 通过mode设定动画的类型&#xff0c;circle为圆圈的形状&#xff0…

探索Spring中的属性注入:@Value注解解析与应用

探索Spring中的属性注入&#xff1a;Value注解解析与应用 探索Spring中的属性注入&#xff1a;Value注解解析与应用摘要引言正文作用代码准备示例注入字符串注入属性注入Bean及其属性 其他属性注入优先级问题对Value属性注入的扩展Spring Boot对Value类型转换的扩展 代码案例演…

【单调栈】[USACO09MAR] Look Up S

题目描述 约翰的 N N N 头奶牛站成一排,奶牛 i i i 的身高是 H i H_i Hi​。现在,每只奶牛都在向右看齐。对于奶牛 i i i,如果奶牛 j j j 满足 i < j i<j i<j 且 H i < H j H_i<H_j Hi​<Hj​,我们可以说奶牛 i i i 可以仰望奶牛 j j j。 求出…