代码随想录刷题day24丨93.复原IP地址 ,78.子集 , 90.子集II

news2025/1/11 12:54:51

代码随想录刷题day24丨93.复原IP地址 ,78.子集 , 90.子集II

1.题目

1.1复原IP地址

  • 题目链接:93. 复原 IP 地址 - 力扣(LeetCode)

    在这里插入图片描述

  • 视频讲解:回溯算法如何分割字符串并判断是合法IP?| LeetCode:93.复原IP地址_哔哩哔哩_bilibili

  • 文档讲解:https://programmercarl.com/0093.%E5%A4%8D%E5%8E%9FIP%E5%9C%B0%E5%9D%80.html

  • 解题思路:回溯

    • 只要意识到这是切割问题,切割问题就可以使用回溯搜索法把所有可能性搜出来

    • 切割问题可以抽象为树型结构,如图:

      在这里插入图片描述

    • 回溯三部曲

      • 递归函数参数

        • startIndex一定是需要的,因为不能重复分割,记录下一层递归分割的起始位置。

        • 需要一个变量pointNum,记录添加逗点的数量。

          List<String> result = new ArrayList<>();
          void backtracking(StringBuilder s,int startIndex,int pointNum)
          
      • 递归终止条件

        • 本题明确要求只会分成4段,所以不能用切割线切到最后作为终止条件,而是分割的段数作为终止条件。

        • pointNum表示逗点数量,pointNum为3说明字符串分成了4段了。

        • 然后验证一下第四段是否合法,如果合法就加入到结果集里

          if(pointNum == 3){
              if(isVaild(s,startIndex,s.length() -1)){
                  result.add(s.toString());
              }
              return;
          }
          
      • 单层搜索的逻辑

        • for (int i = startIndex; i < s.length(); i++)循环中 [startIndex, i] 这个区间就是截取的子串,需要判断这个子串是否合法。

        • 如果合法就在字符串后面加上符号.表示已经分割。

        • 如果不合法就结束本层循环,如图中剪掉的分支:

          在这里插入图片描述

        • 然后就是递归和回溯的过程:

        • 递归调用时,下一层递归的startIndex要从i+2开始(因为需要在字符串中加入了分隔符.),同时记录分割符的数量pointNum 要 +1。

        • 回溯的时候,就将刚刚加入的分隔符. 删掉就可以了,pointNum也要-1。

          for(int i = startIndex;i < s.length();i++){
              if(isVaild(s,startIndex,i)){
                  s.insert(i + 1,'.');
                  pointNum += 1;
                  backtracking(s,i + 2,pointNum);
                  s.deleteCharAt(i + 1);
                  pointNum -= 1;
              }
          }
          
  • 代码:

    class Solution {
        List<String> result = new ArrayList<>();
        public List<String> restoreIpAddresses(String s) {
            StringBuilder sb = new StringBuilder(s);
            backtracking(sb,0,0);
            return result;
        }
    
        void backtracking(StringBuilder s,int startIndex,int pointNum){
            if(pointNum == 3){
                if(isVaild(s,startIndex,s.length() -1)){
                    result.add(s.toString());
                }
                return;
            }
    
            for(int i = startIndex;i < s.length();i++){
                if(isVaild(s,startIndex,i)){
                    s.insert(i + 1,'.');
                    pointNum += 1;
                    backtracking(s,i + 2,pointNum);
                    s.deleteCharAt(i + 1);
                    pointNum -= 1;
                }
            }
        }
    
        // 判断字符串s在左闭⼜闭区间[start, end]所组成的数字是否合法
        private Boolean isVaild(StringBuilder s,int start,int end){
            if(start > end){
                return false;
            }
            if(s.charAt(start) == '0' && start != end){
                return false;
            }
            int num = 0;
            for(int i = start;i <= end;i++){
                int digit = s.charAt(i) - '0';
                num = num * 10 + digit;
                if(num > 255){
                    return false;
                }
            }
            return true;
        }
    }
    
  • 总结:

    • 判断子串是否合法,主要考虑到如下三点:
      • 段位以0为开头的数字不合法
      • 段位里有非正整数字符不合法
      • 段位如果大于255了不合法

1.2子集

  • 题目链接:78. 子集 - 力扣(LeetCode)

    在这里插入图片描述

  • 视频讲解:回溯算法解决子集问题,树上节点都是目标集和! | LeetCode:78.子集_哔哩哔哩_bilibili

  • 文档讲解:https://programmercarl.com/0078.%E5%AD%90%E9%9B%86.html

  • 解题思路:回溯

    • 子集也是一种组合问题,因为它的集合是无序的

    • 那么既然是无序,取过的元素不会重复取,写回溯算法的时候,for就要从startIndex开始,而不是从0开始!

    • 以示例中nums = [1,2,3]为例把求子集抽象为树型结构,如下:

      在这里插入图片描述

    • 从图中红线部分,可以看出遍历这个树的时候,把所有节点都记录下来,就是要求的子集集合

    • 回溯三部曲

      • 递归函数参数

        List<List<Integer>> result = new ArrayList<>();
        List<Integer> path = new ArrayList<>();
        void backtracking(int[] nums,int startIndex)
        
      • 递归终止条件

        if(startIndex >= nums.length){
            return;
        }
        
      • 单层搜索逻辑

        for(int i = startIndex;i < nums.length;i++){
            path.add(nums[i]);
            backtracking(nums,i + 1);
            path.remove(path.size() -1);
        }
        
  • 代码:

    class Solution {
        List<List<Integer>> result = new ArrayList<>();
        List<Integer> path = new ArrayList<>();
        public List<List<Integer>> subsets(int[] nums) {
            backtracking(nums,0);
            return result;
        }
    
        void backtracking(int[] nums,int startIndex){
            result.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.remove(path.size() -1);
            }
        }
    }
    
  • 总结:

    • 什么时候for可以从0开始呢?
      • 求排列问题的时候,就要从0开始

1.3子集II

  • 题目链接:90. 子集 II - 力扣(LeetCode)

    在这里插入图片描述

  • 视频讲解:回溯算法解决子集问题,如何去重?| LeetCode:90.子集II_哔哩哔哩_bilibili

  • 文档讲解:https://programmercarl.com/0090.%E5%AD%90%E9%9B%86II.html

  • 解题思路:回溯

    • 用示例中的[1, 2, 2] 来举例,如图所示:

    在这里插入图片描述

    • 从图中可以看出,同一树层上重复取2 就要过滤掉,同一树枝上就可以重复取2,因为同一树枝上元素的集合才是唯一子集!
  • 代码:

    //法一:使用used
    class Solution {
        List<List<Integer>> result = new ArrayList<>();
        List<Integer> path = new ArrayList<>();
        boolean[] used;
        public List<List<Integer>> subsetsWithDup(int[] nums) {
            if(nums.length == 0){
                result.add(path);
                return result;
            }
            Arrays.sort(nums);
            used = new boolean[nums.length];
            backtracking(nums,0);
            return result;
        }
    
        void backtracking(int[] nums,int startIndex){
            result.add(new ArrayList<>(path));
            if(startIndex >= nums.length){
                return;
            }
    
            for(int i = startIndex;i < nums.length;i++){
                if(i > 0 && nums[i] == nums[i - 1] && used[i - 1] == false){
                    continue;
                }
                path.add(nums[i]);
                used[i] = true;
                backtracking(nums,i + 1);
                path.remove(path.size() -1);
                used[i] = false;
            }
        }
    
    }
    
    //法二:不使用used
    //通过 i > startIndex && nums[i] == nums[i - 1] 来检查是否在同一递归层中遇到相同的元素。条件 i > startIndex 保证了只会在每一层中跳过同一层已经用过的重复元素,而不会跳过递归深层中的重复元素。
    //递归层次概念:每进入一次 backtracking 就相当于进入了一个新的递归层。每个层的 startIndex 是不同的(即在数组中不同的起始位置),因此我们只需要在当前递归层中避免使用相同的元素。
    class Solution {
        List<List<Integer>> result = new ArrayList<>();
        List<Integer> path = new ArrayList<>();
        public List<List<Integer>> subsetsWithDup(int[] nums) {
            if(nums.length == 0){
                result.add(path);
                return result;
            }
            Arrays.sort(nums);
            backtracking(nums,0);
            return result;
        }
    
        void backtracking(int[] nums,int startIndex){
            result.add(new ArrayList<>(path));
            if(startIndex >= nums.length){
                return;
            }
    
            for(int i = startIndex;i < nums.length;i++){
                if(i > startIndex && nums[i] == nums[i - 1]){
                    continue;
                }
                path.add(nums[i]);
                backtracking(nums,i + 1);
                path.remove(path.size() -1);
            }
        }
    
    }
    
  • 总结:

    • 理解“树层去重”和“树枝去重”非常重要

    • 注意去重需要先对集合排序

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

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

相关文章

2024年高教杯国赛(B题)数学建模竞赛解题思路|完整代码论文集合

我是Tina表姐&#xff0c;毕业于中国人民大学&#xff0c;对数学建模的热爱让我在这一领域深耕多年。我的建模思路已经帮助了百余位学习者和参赛者在数学建模的道路上取得了显著的进步和成就。现在&#xff0c;我将这份宝贵的经验和知识凝练成一份全面的解题思路与代码论文集合…

【2024高教社杯国赛A题】数学建模国赛建模过程+完整代码论文全解全析

你是否在寻找数学建模比赛的突破点&#xff1f;数学建模进阶思路&#xff01; 作为经验丰富的数学建模团队&#xff0c;我们将为你带来2024国赛数学建模竞赛&#xff08;A题&#xff09;的全面解析。这个解决方案包不仅包括完整的代码实现&#xff0c;还有详尽的建模过程和解析…

WebAPI(二)、DOM事件监听、事件对象event、事件流、事件委托、页面加载与滚动事件、页面尺寸事件

文章目录 一、 DOM事件1. 事件监听2. 事件类型(1)、鼠标事件(2)、焦点事件(3)、键盘事件(4)、文本事件 3. 事件对象(1)、获取事件对象(2)、事件对象常用属性 4. 环境对象 this5. 回调函数 二、 DOM事件进阶1. 事件流(1)、 捕获阶段(2)、 冒泡阶段(3)、 阻止冒泡(4) 、阻止元素默…

Optuna发布 4.0 重大更新:多目标TPESampler自动化超参数优化速度提升显著

Optuna这个备受欢迎的超参数优化框架在近期发布了其第四个主要版本。自2018年首次亮相以来&#xff0c;Optuna不断发展&#xff0c;现已成为机器学习领域的重要工具。其用户社区持续壮大&#xff0c;目前已达到以下里程碑&#xff1a; 10,000 GitHub星标每月300万 下载量16,00…

静心是良性循环

我发现一个挺有意思的事&#xff1a; 花时间修炼自己的心&#xff0c;让自己静心&#xff0c;就愿意多花时间修炼&#xff0c;会更加静心&#xff0c;这样就良性循环了。难的是刚开始&#xff0c;心不静&#xff0c;没时间修炼&#xff0c;心静不下来&#xff0c;更没时间修炼&…

分类中的语义一致性约束:助力模型优化

前言 这里介绍一篇笔者在去年ACL上发表的一篇文章&#xff0c;使用了空间语义约束来提高多模态分类的效果&#xff0c;类似的思路笔者也在视频描述等方向进行了尝试&#xff0c;也都取得了不错的效果。这种建模时对特征进行有意义的划分和约束对模型还是很有帮助的&#xff0c;…

基于人工智能的文本摘要生成系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 文本摘要生成是自然语言处理&#xff08;NLP&#xff09;中的一个重要任务&#xff0c;它旨在从长文本中提取出核心内容&#xff0c;生…

【学习笔记】SSL证书安全机制之证书验证

前言&#xff1a;每当Client从Server收到一张证书&#xff0c;有2件事Client需要去验证&#xff1a; 证书是否有效&#xff1f; 证书只是文件中的文本Client如何知道内容能够信任&#xff1f;Server是否是证书真正的拥有者&#xff1f; 证书可以公开获取Client如何知道Server是…

OM5光纤:优化您的数据中心性能

随着数据中心需求的持续增长&#xff0c;传统OM3和OM4光纤在传输距离和带宽方面的局限性日益显现。为应对这些挑战&#xff0c;OM5光纤应运而生。OM5光纤又称为宽带多模光纤&#xff08;WBMMF&#xff09;&#xff0c;通过支持单根光纤上的多波长传输&#xff0c;显著提高了数据…

ubuntu linux搭建lvgl v9

记录一下ubuntu linux搭建 lvgl的过程 本地环境:ubuntu 16.04 ubuntu lvgl sdl2 1 获取源码 git clone https://github.com/lvgl/lv_port_linux.git cd lv_port_linux/ git submodule update --init --recursive查看分支: git branch -a 我选择的是9.2(master分支一直在变动…

无人机执照拿到后怎么就业?方向有哪些?就业率如何?

无人机执照拿到后&#xff0c;就业方向广泛且多样&#xff0c;就业率也呈现出逐年上升的趋势。这主要得益于无人机技术的广泛应用和无人机市场的不断扩大。以下是对无人机执照持有者就业情况的详细分析&#xff1a; 就业方向 1. 无人机飞行操作&#xff1a; 无人机飞手可以从…

34465A-61/2 数字万用表(六位半)

34465A-61/2 数字万用表(六位半) 文章目录 34465A-61/2 数字万用表(六位半)前言一、测DC/AC电压二、测DC/AC电流四、测电阻五、测电容六、测二极管七、保存截图流程前言 1、6位半数字万用表通常具有200,000个计数器,可以显示最大为199999的数值。相比普通数字万用表,6位半…

经典文献阅读之--Representing 3D sparse map points....(用于相机重定位的3D点线稀疏地图)

0.简介 最近在视觉定位和地图制图方面取得了显著的进展&#xff0c;成功地将点特征和线特征进行了整合。然而&#xff0c;将定位框架扩展到包括额外的地图组件往往会导致对匹配任务的内存和计算资源需求增加。《Representing 3D sparse map points and lines for camera reloc…

尝试用java spring boot+VUE3实现前后端分离部署(8/31)

前言 这几天开学了&#xff0c;公司这边几个和学校对接的项目都挺忙的&#xff0c;然后我又开始有点闲的情况了。问大佬能不能继续看看若依的项目&#xff0c;大佬让我自己去学了。在看若依的项目的时候在想&#xff0c;python的FLASK后端实现和JAVA spring boot的实现差别大不…

通过指令微调提升语言模型性能

人工智能咨询培训老师叶梓 转载标明出处 如何让机器更好地理解和执行人类的指令一直是一个重要课题。Google的研究团队中提出了一种新的方法&#xff0c;通过指令微调&#xff08;instruction finetuning&#xff09;来提升语言模型的性能和泛化能力。 他们主要研究了如何通过…

美食|基于SpringBoot+vue的美食网站(源码+数据库+文档)

美食网站 基于SSMvue的美食网站 一、前言 二、系统设计 三、系统功能设计 系统功能实现 后台模块实现 管理员模块实现 用户模块实现 餐厅模块实现 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 博主介绍&…

JavaScript拷贝的艺术:玩转深拷贝和浅拷贝

前言 在实际的项目开发中&#xff0c;我们时刻都在使用数据拷贝功能&#xff0c;赋值、深拷贝和浅拷贝是前端开发中常见的概念&#xff0c;用于复制简单数据类型&#xff08;字符串、数值、布尔值&#xff09;和引用类型&#xff08;对象、数组&#xff09;。它们的主要区别在…

第十六篇:走入计算机网络的传输层--传输层概述

1. 传输层的功能 ① 分割与重组数据 一次数据传输有大小限制&#xff0c;传输层需要做数据分割&#xff0c;所以在数据送达后必然也需要做数据重组。 ② 按端口号寻址 IP只能定位数据哪台主机&#xff0c;无法判断数据报文应该交给哪个应用&#xff0c;传输层给每个应用都设…

中小型局域网组网规划与实施

一、绪论 1.1 背景 本课题以中小型企业网络搭建为背景&#xff0c;实现网络规划与设计和模拟。该企业网有四个部门&#xff0c;人力部、研发部、市场部和财务部&#xff0c;不同部门分别划分VLAN&#xff0c;不同VLAN之间分配不同的IP地址段。内外网之间要互通。 1.2 发展趋势…

关于前端分辨率兼容和显示器缩放兼容的处理

如下图所示&#xff0c;我们的电脑屏幕可以进行缩放&#xff0c;和分辨率的切换。 我们在项目开发中&#xff0c;时常需要适配不同的分辨率。 一般来说&#xff0c;开发人员电脑分辨率显示正常的页面&#xff0c;只会在更小的分辨率尺寸中出现问题。 所以当测试人员给我们提分…