刷题计划day24 回溯(三)【复原 IP 地址】【子集】【子集 II】
⚡刷题计划day24 回溯(三)继续,回溯一共会有五个专题,敬请期待关注,可以点个免费的赞哦~
目录
题目一:复原 IP 地址
回溯三部曲
题目二:78. 子集
回溯三部曲
题目三:90. 子集 II
题目一:复原 IP 地址
93. 复原 IP 地址
(https://leetcode.cn/problems/restore-ip-addresses/description/)
刚看到这道题目可能还会有些茫然,
其实只要意识到这是切割问题,切割问题就可以使用回溯搜索法把所有可能性搜出来,和刚做过的131.分割回文串 就十分类似了。
切割问题可以抽象为树型结构,如图:

回溯三部曲
1.递归参数
startIndex一定是需要的,因为不能重复分割,记录下一层递归分割的起始位置。
本题我们还需要一个变量pointNum,记录添加逗点的数量。
List<String> res = new ArrayList<>();
void backtracking(StringBuilder sb,int startIndex,int pointNum)
2.递归终止条件
本题明确要求只会分成4段,所以不能用切割线切到最后作为终止条件,而是分割的段数作为终止条件。
pointNum表示逗点数量,pointNum为3说明字符串分成了4段了。
然后验证一下第四段是否合法,如果合法就加入到结果集里
代码如下:
if(pointNum==3){if(isValid(sb,startIndex,sb.length()-1)){res.add(sb.toString());}return;
}
3.单层搜索的逻辑
在for (int i = startIndex; i < s.size(); i++)循环中 [startIndex, i] 这个区间就是截取的子串,需要判断这个子串是否合法。
如果合法就在字符串后面加上符号.表示已经分割。
如果不合法就结束本层循环,如图中剪掉的分支:

然后就是递归和回溯的过程:
递归调用时,下一层递归的startIndex要从i+2开始(因为需要在字符串中加入了分隔符.),同时记录分割符的数量pointNum 要 +1。
回溯的时候,就将刚刚加入的分隔符. 删掉就可以了。
for(int i=startIndex;i<sb.length();i++){if(isValid(sb,startIndex,i)){sb.insert(i+1,'.');backtracking(sb,i+2,pointNum+1);sb.deleteCharAt(i+1);}else {break;}
}
整体代码如下:
class Solution {List<String> res = new ArrayList<>();public List<String> restoreIpAddresses(String s) {StringBuilder sb = new StringBuilder(s);backtracking(sb,0,0);return res;
}// startIndex: 搜索的起始位置, pointNum:添加逗点的数量public void backtracking(StringBuilder sb,int startIndex,int pointNum){if(pointNum==3){if(isValid(sb,startIndex,sb.length()-1)){res.add(sb.toString());}return;}for(int i=startIndex;i<sb.length();i++){if(isValid(sb,startIndex,i)){sb.insert(i+1,'.');//在str的后⾯插⼊⼀个逗点backtracking(sb,i+2,pointNum+1);// 插⼊逗点之后下⼀个⼦串的起始位置为i+2sb.deleteCharAt(i+1);// 回溯删掉逗点}else {break;}}}// 判断字符串s在左闭⼜闭区间[start, end]所组成的数字是否合法public boolean isValid(StringBuilder sb,int start,int end){if(start>end){return false;}// 0开头的数字不合法if(sb.charAt(start)=='0' && start!=end){return false;}// 如果⼤于255了不合法int num=0;for(int i=start;i<=end;i++){int digit = sb.charAt(i)-'0';num = num*10+digit;if(num>255){return false;}}return true;}
}
题目二:78. 子集
78. 子集
(https://leetcode.cn/problems/subsets/description/)
求子集问题和之前的组合问题,分割问题又略有区别,但是也在我们回溯第一节中给出的模板里的
如果把 子集问题、组合问题、分割问题都抽象为一棵树的话,那么组合问题和分割问题都是收集树的叶子节点,而子集问题是找树的所有节点!
其实子集也是一种组合问题,因为它的集合是无序的,子集{1,2} 和 子集{2,1}是一样的。
那么既然是无序,取过的元素不会重复取,写回溯算法的时候,for就要从startIndex开始,而不是从0开始!
有同学问了,什么时候for可以从0开始呢?
求排列问题的时候,就要从0开始,因为集合是有序的,{1, 2} 和{2, 1}是两个集合,排列问题我们后续的文章就会讲到的。
以示例中nums = [1,2,3]为例把求子集抽象为树型结构,如下:

回溯三部曲
1.递归函数参数
需要path收集符合条件路径,res存放符合的path。
List<List<Integer>> res = new ArrayList<>();LinkedList<Integer> path = new LinkedList<>();
然后参数需要传入nums,startIndex。
void backtracking(int[] nums,int startIndex)
2.递归终止条件
如图:

我们可以发现,当剩余集合为空的时候,也是叶子节点,便是终止的条件。
怎么判断呢,当startIndex大于数组的长度便表示终止了,后续也没有元素课取了。
if(startIndex>=nums.length){return;
}
其实可以不需要加终止条件,因为startIndex >= nums.size(),本层for循环本来也结束了。
3.单层搜索逻辑
求取子集问题,不需要任何剪枝!因为子集就是要遍历整棵树。
那么单层递归逻辑代码如下:
for(int i = startIndex;i<nums.length;i++){path.add(nums[i]);backtracking(nums,i+1);path.removeLast();
}
整体代码如下:
class Solution {List<List<Integer>> res = new ArrayList<>();LinkedList<Integer> path = new LinkedList<>();
public List<List<Integer>> subsets(int[] nums) {backtracking(nums,0);return res;}
public void backtracking(int[] nums,int startIndex){res.add(new ArrayList<>(path));if(startIndex>=nums.length){return;}
for(int i = startIndex;i<nums.length;i++){path.add(nums[i]);backtracking(nums,i+1);path.removeLast();}}
}
题目三:90. 子集 II
90. 子集 II
(https://leetcode.cn/problems/subsets-ii/description/)
此题与上一题的区别就是集合里有重复元素了,而且求取的子集要去重。
然后关于回溯算法中的去重问题,在上次的章节中已经讲解过,
关于去重,理解好"树层去重","树枝去重",就可以解决了。
用示例中的[1, 2, 2] 来举例,如图所示: (注意去重需要先对集合排序)

从图中可以看出,同一树层上重复取2 就要过滤掉,同一树枝上就可以重复取2,因为同一树枝上元素的集合才是唯一子集!
完整代码如下:
class Solution {List<List<Integer>> res = new ArrayList<>();LinkedList<Integer> path = new LinkedList<>();public List<List<Integer>> subsetsWithDup(int[] nums) {Arrays.sort(nums);backtrcking(nums,0);return res;}public void backtrcking(int[] nums,int startIndex){res.add(new ArrayList<>(path));
for(int i=startIndex;i<nums.length;i++){//去重if(i>startIndex && nums[i]==nums[i-1]){continue;}path.add(nums[i]);backtrcking(nums,i+1);path.removeLast();}}
}
相关文章:
刷题计划day24 回溯(三)【复原 IP 地址】【子集】【子集 II】
⚡刷题计划day24 回溯(三)继续,回溯一共会有五个专题,敬请期待关注,可以点个免费的赞哦~ 往期可看专栏,关注不迷路, 您的支持是我的最大动力🌹~ 目录 题目一:复原 IP…...
从“找三角形”讲“等腰三角形”
【题目】 周长为11,且各边长均为整数的三角形有哪些? 【答案】 四种,边长分别为: 2 4 5 3 3 5 1 5 5 3 4 4 【解析】 讲解等腰三角形的概念时,传统方法一般向学生展示一个等腰三角形的实物模型,这…...
Java中的泛型方法和泛型类
在Java编程语言中,泛型(Generics)是一个强大的特性,它使得类、接口和方法能够灵活地操作各种数据类型,同时保持类型安全。泛型主要通过类型参数(Type Parameters)来实现,这些类型参数…...
springboot学习-spring-boot-data-jdbc分页/排序/多表查询的例子
上次使用的是JdbcTemplate实现的,是比较老的方式,重新用spring boot data jdbc和jdbc client 实现一遍。也比较一下这几种的编码差异。数据库方面JAVA给了太多选择,反而不好选了。 上次就试图直接用: public interface UserRepo…...
通信与网络基础
1.网络通信基本概念 通信:人、物通过某种介质和行为进行信息传递与交流 网络通信:终端设备之间通过计算机网络进行通信 两个终端通过网线传递文件 多个终端通过路由器传递文件 终端通过Internet下载文件 2.信息传递过程 图1-1 假定A计算机访问B的web…...
【3.存储系统】综合大题
【考点】存储系统综合大题 【2011年408真题】某计算机存储器按字节编址,虚拟(逻辑)地址空间大小为16 MB,主存(物理)地址空间大小为1 MB,页面大小为4 KB;Cache采用直接映射方式,共8行;主存与Cache之间交换的…...
【Linux】【字符设备驱动】深入解析
Linux字符设备驱动程序用于控制不支持随机访问的硬件设备,如串行端口、打印机、调制解调器等。这类设备通常以字符流的形式与用户空间程序进行交互。本节将深入探讨字符设备驱动的设计原理、实现细节及其与内核其他组件的交互。 1. 引言 字符设备驱动程序是Linux内…...
【JavaEE】多线程(2)
一、线程安全 1.1 线程安全的概念 线程是随机调度执行的,如果多线程环境下的程序运行的结果符合我们预期则说明线程安全,反之,如果遇到其他结果甚至引起了bug则说明线程不安全 1.2 经典例子与解释 下面举一个经典的线程不安全的例子&…...
mac下Gpt Chrome升级成GptBrowser书签和保存的密码恢复
cd /Users/自己的用户名/Library/Application\ Support/ 目录下有 GPT\ Chrome/ Google/ GptBrowser/ GPT\ Chrome 为原来的chrome浏览器的文件存储目录. GptBrowser 为升级后chrome浏览器存储目录 书签所在的文件 Bookmarks 登录账号Login 相关的文件 拷贝到GptBrow…...
使用Grafana K6来测测你的系统负载能力
背景 近期我们有个号称会有很高很高并发的系统要上线,为了测试一下自己开发的系统的负载能力,准备了点海克斯科技,来看看抗不抗的住。 之前笔者写过用Apache JMeter进行压力测试的文章(传送门👉:https://…...
【论文复现】基于BERT的语义分析实现
📝个人主页🌹:Eternity._ 🌹🌹期待您的关注 🌹🌹 ❀ WRN: 宽度残差网络 概述语义分类文本分类情感分类 实现原理 核心逻辑pre_deal.pytrain.pytest_demo.py 实现方式&演示效果训练阶段测试阶…...
CTF-RE: STL逆向 [NewStarCTF 2023 公开赛道 STL] WP
多看看STL题就会了,很简单 int __fastcall main(int argc, const char **argv, const char **envp) {__int64 v3; // rbx__int64 v4; // raxchar v5; // bl_BYTE *v6; // rax_QWORD *v7; // rax__int64 v8; // rax__int64 v9; // raxint i; // [rsp0h] [rbp-250h]int j; // [r…...
实习冲刺第三十六天
46.全排列 给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 示例 1: 输入:nums [1,2,3] 输出:[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]示例 2: 输入&#…...
【Zemax光学设计实训三】---激光缩束镜的设计优化
前言与目录 技术设计要求: 设计一个激光扩束镜,使用的波长为1064nm,输入光束直径为10mm,输出光束的直径为2mm,且输入光束和输出光束平行(即平行光入射,平行光出射)。要求只使用两片…...
TCP/IP协议簇自学笔记
摘抄于大学期间记录在QQ空间的一篇自学笔记,当前清理空间,本来想直接删除掉的,但是感觉有些舍不得,因此先搬移过来。 曾经,我只知道socket函数能进行网络间数据的通信,知道tcp/ip协议也是用来进行网络数据…...
Spring Boot教程之十一:获取Request 请求 和 Put请求
如何在 Spring Boot 中获取Request Body? Java 语言是所有编程语言中最流行的语言之一。使用 Java 编程语言有几个优点,无论是出于安全目的还是构建大型分发项目。使用 Java 的优点之一是 Java 试图借助类、继承、多态等概念将语言中的每个概念与现实世…...
计算机网络(二)
ip地址:11010010:01011110:00100100:00010100 子网掩码:11111111:11111111:11111111:11000000 and :11010010:01011110:00100100:00000000 210.94.36.0的下一站为R1 因为255为11111111 192为ÿ…...
如何在Python中进行数学建模?
数学建模是数据科学中使用的强大工具,通过数学方程和算法来表示真实世界的系统和现象。Python拥有丰富的库生态系统,为开发和实现数学模型提供了一个很好的平台。本文将指导您完成Python中的数学建模过程,重点关注数据科学中的应用。 数学建…...
JavaSE——类与对象(5)
一、抽象类 1.1为什么需要抽象类 父类的某些方法,不确定怎么实现,也不需要实现。 class Animal{public String name;public Animal(String name){this.name name;}public void eat()//这里实现了也没有意义{System.out.println("这是一个动物&am…...
Istio笔记01--快速体验Istio
Istio笔记01--快速体验Istio 介绍部署与测试部署k8s安装istio测试istio 注意事项说明 介绍 Istio是当前最热门的服务网格产品,已经被广泛应用于各个云厂商和IT互联网公司。企业可以基于Istio轻松构建服务网格,在接入过程中应用代码无需更改,…...
Java多线程:从入门到进阶
Java多线程:从入门到进阶 1. 引入:为什么需要多线程? 1.1 单线程的瓶颈 假设你要下载三个文件,单线程的做法是:一个个下载,总时间 文件1 文件2 文件3。 downloadFile1(); // 等待完成 downloadFile2();…...
终极指南:Awoo Installer - Nintendo Switch游戏安装的免费开源解决方案
终极指南:Awoo Installer - Nintendo Switch游戏安装的免费开源解决方案 【免费下载链接】Awoo-Installer A No-Bullshit NSP, NSZ, XCI, and XCZ Installer for Nintendo Switch 项目地址: https://gitcode.com/gh_mirrors/aw/Awoo-Installer 还在为Switch游…...
【STM32F407启动探秘】从复位向量到main():深入剖析启动文件与BOOT模式
1. STM32F407启动过程全景图 当你按下STM32F407开发板的电源按钮时,芯片内部就像被施了魔法一样开始运转。这个看似简单的上电过程,实际上隐藏着一套精密的启动机制。作为开发者,理解这个过程就像掌握了一把打开STM32内核奥秘的钥匙。 我刚开…...
智慧交通系统安全漏洞深度解析:从明文传输到固件攻击的防御启示
1. 项目概述:一次对智慧交通“神经末梢”的深度安全审视2014年的DEF CON黑客大会,向来是安全研究的风向标。那一年,IOActive的首席技术官Cesar Cerrudo在台上展示的,不是某个炫酷的软件漏洞,而是一个关于我们每天经过的…...
告别环境配置噩梦:用Shell脚本一键搞定VCS与Verdi的联调环境
芯片验证工程师的效率革命:Shell脚本全自动构建VCSVerdi联调环境 每次开始新项目都要重复配置验证环境?还在为VCS编译选项和Verdi波形调试的手动操作浪费时间?资深验证工程师的日常,不该被这些重复劳动占据。本文将带你用Shell脚本…...
基于PyTorch的图像分类实战:从数据增强到模型微调全流程解析
1. 项目概述:一个基于深度学习的开源图像识别工具最近在整理个人项目库时,翻到了一个挺有意思的仓库,叫jyao97/xylocopa。乍一看这个名字,可能有点摸不着头脑,但如果你对昆虫学或者开源项目命名有点了解,就…...
线性码电路优化:从理论到硬件实现
1. 线性码与电路合成基础线性码在数字通信和存储系统中扮演着至关重要的角色,它通过在原始数据中添加冗余信息来实现错误检测和纠正。这种编码方式的核心数学原理基于有限域上的线性代数运算,使得编码和解码过程可以通过高效的矩阵运算实现。在硬件实现层…...
code-outline:为AI编程助手设计的代码结构导航工具,节省90% Token消耗
1. 项目概述:为AI编程助手打造的代码结构导航仪如果你和我一样,日常开发中重度依赖像Claude Code、Cursor Agent或者Aider这类AI编程助手,那你肯定遇到过这个痛点:想让AI帮你理解一个陌生项目,或者修改一个大型文件里的…...
手把手教你搞定产品EMC静电放电测试:从PCB布局到TVS选型的完整避坑指南
手把手教你搞定产品EMC静电放电测试:从PCB布局到TVS选型的完整避坑指南 静电放电(ESD)是电子设备最常见的电磁兼容问题之一。去年某智能家居厂商因ESD测试失败导致产品召回,直接损失超过2000万。这并非孤例——行业数据显示&…...
中间件与依赖系统:构建高效 Web 后端的双重利器
文章目录一、 中间件(Middleware):全局的“拦截器”1.1 核心概念1.2 执行原理1.3 代码实现1.4 多中间件执行顺序二、 依赖系统(Dependency Injection):精细化的“业务注入”2.1 为什么要用依赖系统…...
