代码随想录Day28:回溯算法Part4

Leetcode 93. 复原IP地址

讲解前:

这道题其实在做完切割回文串之后,学会了使用切割的方法来找到字符串的possible 子串之后,思路就会很快找到,细想一下其实无非也就是对given string然后进行切割,只是深度是固定的因为ip只能有四组数字组成,也就是四个sub string,然后呢我们就需要找到所有的possible组合,并且确保组合中每一个substring是符合规则的

并且当我们发现在for循环中,这一整个循环的加入的第一个path都是0开头的长度大于1的substring 的时候,就可以break 整个loop,但是按照我目前的思路,这道题还是有太多细节需要处理,所以我觉得先看题解

讲解后:
 
class Solution:
    def restoreIpAddresses(self, s: str) -> List[str]:
        res = []
        path = []

        # helper function to check if the current substring
        # is a valid ip address
        def is_valid(string):
            if not string:
                return False

            if string[0] == '0' and len(string) > 1:
                return False
            elif int(string) > 255:
                return False
            else:
                return True


        def backtracking(s, res, path, start_index, count):
            # base case
            # if we already added three substring to path
            # check the remaining one is valid or not
            print(path, '-', count)
            if count == 3:
                last_string = s[start_index:]
                if is_valid(last_string):
                    path.append(last_string)
                    res.append('.'.join(path[:]))
                    path.pop()
                    count = count - 1
                return

            
            # recursive case
            for i in range(start_index, len(s)):
                cur_string = s[start_index: i + 1]
                print(cur_string)

                
                if is_valid(cur_string):
                    path.append(cur_string)
                    count = count + 1
                    backtracking(s, res, path, i + 1, count)

                    path.pop()
                    count = count - 1

        backtracking(s, res, path, 0, 0)
        return res 

听完了卡哥的讲解之后发现其实这道题没有那么复杂,关键就在于当对substring是否是一个valid的id进行检查的时候,既然检查的条件会比较多比较繁琐,那么就直接封装成一个函数就好了,然后呢思路就和其他切割字符串的题没区别了,就是检查每一个substring,如果符合要求,就在基础上继续递归找下一个符合条件的,唯一比较特殊的是,在base case中,这里我们只有三条切割线,最后第四个substring需要我们在找完三次之后直接取剩余的string作为第四个,所以在base case中我们还需要再对path进行一次添加,同时这里要注意,这一次添加也需要做回溯操作也就是对path进行pop以及减少count的值


Leetcode 78. 子集

讲解前:

这道题的思路其实很简单,题目中需要求的子集其实我们同样把树形结构画出来之后就发现,只不过是在递归的过程中把path的值收集一下,因为如果我们没有特定的组合的要求,那么每一个path在递归的过程中就是一个组合,并且通过我们对i的递增,还可以保证不会重复

class Solution:
    def subsets(self, nums: List[int]) -> List[List[int]]:
        res = []
        res.append([])
        path = []

        def backtracking(nums, res, path, start_index):

            for i in range(start_index, len(nums)):
                path.append(nums[i])
                res.append(path[:])

                backtracking(nums, res, path, i + 1)

                path.pop()
        
        backtracking(nums, res, path, 0)
        return res 
讲解后:

看了卡哥的讲解发现我其实没必要在for循环里面直接加入res,并且在一开始把空列表先加进去,因为我一开始害怕我们进入函数之后,在重新从2开始递归之前还会出现一次path为空的情况,包括3也是,但其实并不会,在pop了之后,我们还是继续循环并且把数字加入到path之后才会再递归,所以把加入res的代码写在backtracking的第一行就可以了

class Solution:
    def subsets(self, nums: List[int]) -> List[List[int]]:
        res = []
        path = []

        def backtracking(nums, res, path, start_index):
            # every recursive call, add the current path to res
            res.append(path[:])
            
            for i in range(start_index, len(nums)):
                path.append(nums[i])

                backtracking(nums, res, path, i + 1)

                path.pop()
        
        backtracking(nums, res, path, 0)
        return res 

Leetcode 90. 子集II

讲解前:

这道题其实就是之前组合问题的变种,关键就在于去重,就如下图中所示的

[1, 2] 和 [2] 会反复出现,但是我们不能重复取,这里其实思路就是如果我们是在纵向搜索,也就是说是在继续增加数字然后扩大子集,那么就没关系,但是如果是横向的代表集合由前一个数字开头的已经全部找完了,那么如果我们再从这里开始并且当前这个数字和之前的那个一样,那么我们一定会再end up with the same set,那么这里其实还有个更简单的方法如果我们当前的index和上一位相等并且是在i比startindex大的情况下,因为这时候证明我们是在纵向递归,并且要记得这种方法的前提是我们要先把nums变成递增顺序才可以,这样两个相同的数字才会next to each other

class Solution:
    def subsetsWithDup(self, nums: List[int]) -> List[List[int]]:
        res = []
        path = []
        nums.sort()

        def backtracking(nums, res, path, start_index):
            # add the current path to res each recursion
            res.append(path[:])

            for i in range(start_index, len(nums)):
                if i > 0 and nums[i] == nums[i - 1] and i > start_index:
                    continue
                
                path.append(nums[i])
                backtracking(nums, res, path, i + 1)
                path.pop()
        
        backtracking(nums, res, path, 0)
        return res 
讲解后:

卡哥的解法就和之前讲解组合去重问题时候一样,利用了一个used数组,思路依然很清晰,但是这里我的写法更加简便一点,虽然需要在理解了used数组的解法之上去想明白才能看懂

这里是文字解法中利用used数组的解法

class Solution:
    def subsetsWithDup(self, nums):
        result = []
        path = []
        used = [False] * len(nums)
        nums.sort()  # 去重需要排序
        self.backtracking(nums, 0, used, path, result)
        return result

    def backtracking(self, nums, startIndex, used, path, result):
        result.append(path[:])  # 收集子集
        for i in range(startIndex, len(nums)):
            # used[i - 1] == True,说明同一树枝 nums[i - 1] 使用过
            # used[i - 1] == False,说明同一树层 nums[i - 1] 使用过
            # 而我们要对同一树层使用过的元素进行跳过
            if i > 0 and nums[i] == nums[i - 1] and not used[i - 1]:
                continue
            path.append(nums[i])
            used[i] = True
            self.backtracking(nums, i + 1, used, path, result)
            used[i] = False
            path.pop()

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

GLP-1药物固相合成法-载体树脂及层析填料

摘要:在生物医药GLP-1药物制备领域不仅可提供高稳定性载体树脂,还可根据客户需求,合成定制化载体(如预接氨基酸固相合成载体、特殊溶胀度或基团负载量的载体、负载特殊基团的载体、清除树脂等)。同时,海普专…

深入剖析主机安全中的零信任机制及其实施原理

引言 在数字化转型加速与云端服务普及的大背景下,传统依赖边界的网络安全模式逐渐显露出其局限性。面对愈发复杂多变的威胁环境,零信任安全架构作为新一代的安全范式应运而生,尤其是在主机层面的安全实践中,零信任机制正扮演着至…

docker安装wekan

安装mongodb 注意这里用端口映射方法将db的端口映射到宿主机。并且注意自己的映射目录,如果不需要映射目录直接删除-v /home/data/project/wekan/wekandb/db:/data/db -v /home/data/project/wekan/wekandb/configdb:/data/configdb sudo docker run -d --name we…

Golang | Leetcode Golang题解之第7题整数反转

题目&#xff1a; 题解&#xff1a; func reverse(x int) (rev int) {for x ! 0 {if rev < math.MinInt32/10 || rev > math.MaxInt32/10 {return 0}digit : x % 10x / 10rev rev*10 digit}return }

第十三届蓝桥杯大赛软件赛省赛CC++大学B组

第十三届蓝桥杯大赛软件赛省赛CC 大学 B 组 文章目录 第十三届蓝桥杯大赛软件赛省赛CC 大学 B 组1、九进制转十进制2、顺子日期3、刷题统计4、修建灌木5、x进制减法6、统计子矩阵7、积木画8、扫雷9、李白打酒加强版10、砍竹子 1、九进制转十进制 计算器计算即可。2999292。 2、…

如何在 Visual Studio for Mac 中使用 .NET 8 上的 FastReport Avalonia

FastReport Business Graphics .NET&#xff0c;是一款基于fastreport报表开发控件的商业图形库&#xff0c;借助 FastReport 商业图形库&#xff0c;您可以可视化不同的分层数据&#xff0c;构建业务图表以进行进一步分析和决策。利用数据呈现领域专家针对 .NET 7、.NET Core、…

阿里云服务器租用价格,2024年新版活动报价及租用收费标准

2024年阿里云服务器租用费用&#xff0c;云服务器ECS经济型e实例2核2G、3M固定带宽99元一年&#xff0c;轻量应用服务器2核2G3M带宽轻量服务器一年61元&#xff0c;ECS u1服务器2核4G5M固定带宽199元一年&#xff0c;2核4G4M带宽轻量服务器一年165元12个月&#xff0c;2核4G服务…

解决el-table设置固定高度后,展示不同列时表格高度变小bug

解决el-table设置固定高度后&#xff0c;展示不同列时表格高度变小bug 1、需求分析2、解决方案 1、需求分析 在el-table使用过程中&#xff0c;选择多个参数展示更多列时会出现高度变小问题究其原因可知是el-table列动态发生变化后&#xff0c;el-table__body-wrapper的高度变…

HTML - 你是如何理解link和@import的

难度级别:初级及以上 提问概率:55% link是我们非常熟悉的一个HTML标签,用于引入CSS文件,而@import则存在于CSS文件内部,用于再次引入其他的CSS文件。所以很显然,执行顺序上,link标签会随着HTML文档加载,开始触发下载CSS文件的操作。…

clickhouse 源码编译部署

clickhouse 源码编译部署 版本 21.7.9.7 点击build project&#xff0c;编译工程&#xff0c;经过一定时间&#xff08;第一次编译可能几个小时&#xff0c;后续再编译&#xff0c;只编译有改动的文件&#xff09;生成release目录 在cmake-build-release → programs目录下…

【机器学习】科学库使用第3篇:机器学习概述,学习目标【附代码文档】

机器学习&#xff08;科学计算库&#xff09;完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;机器学习&#xff08;常用科学计算库的使用&#xff09;基础定位、目标&#xff0c;机器学习概述定位,目标,学习目标,学习目标,1 人工智能应用场景,2 人工智能小…

【generate】如何维护一套icon组件库,直接输出svg为react component

https://github.com/ant-design/ant-design-web3/pull/761/files 实现了icon-preview(通过jsdoc, 鼠标放在组件上可以看到icon的样式)&#xff0c;因为打包方式、产物以及命名上有一些不同&#xff0c;可能需要稍加改造。 这个同步脚本应该后续也用得上&#xff0c;略加改造同步…

git分支-基本分支与合并

问题假设 让我们通过一个简单的分支和合并的例子&#xff0c;演示在实际工作中可能会使用的工作流程。将按照以下步骤进行&#xff1a; 在网站上进行一些工作。为正在开发的新用户故事创建一个分支。在该分支上进行一些工作。 在这个阶段&#xff0c;我们可能会接到一个电话…

原型链污染攻击也称JavaScript Prototype 污染攻击

JavaScript数据类型 let和var关键字的区别 使用var或let关键字可以定义变量 let和var的区别如下&#xff1a; var是全局作用域&#xff0c;let 只在当前代码块内有效 当在代码块外访问let声明的变量时会报错 var有变量提升&#xff0c;let没有变量提升 let必须先声明…

U-net在乳腺癌医学图像分割方面的应用

图像几何变换和生成对抗网络 通过旋转、反转、剪切图像对乳腺医学图像进行数据增强之后&#xff0c;可以提高模型的准确性。但是当前简单的分割和几何变换在医疗图像数据中不会简单使用&#xff0c;而是会集合生成对抗网络&#xff08;GAN&#xff09;结合使用。使用生成对抗网…

zookeeper监听集群节点的实现zkclient组件实现方案(Java版)

ZooKeeper Watcher 机制 client 向zookeeper 注册监听client注册的同时会存储一个WatchManager对象向zookeeper发生改变则notification client 并发送一个WatchManager对象,然后client再更新该对象 package com.jacky.zk.demo;import org.I0Itec.zkclient.IZkChildListener;…

https访问http的minio 图片展示不出来

问题描述&#xff1a;请求到的图片地址单独访问能显示&#xff0c;但是在网页中展示不出来 原因&#xff1a;https中直接访问http是不行的&#xff0c;需要用nginx再转发一下 nginx配置如下&#xff08;注意&#xff1a;9000是minio默认端口&#xff0c;已经占用&#xff0c;…

基于Spring Boot的餐厅点餐系统

基于Spring Boot的餐厅点餐系统 开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#xff1a;Maven3.3.9 部分系统展示 管理员登录界面 用户注册登录界面 …

MySQL-视图:视图概述、创建、查看、更新、修改、删除

第14章 视图 1. 常见的数据库对象2. 视图概述2.1 为什么使用视图&#xff1f;2.2 视图的理解 3. 创建视图3.1 创建单表视图3.2 创建多表联合视图3.3 基于视图创建视图 4. 查看视图5. 更新视图的数据5.1 一般情况5.2 不可更新的视图 6. 修改、删除视图6.1 修改视图6.2 删除视图 …

uniapp微信小程序真机图片不显示

不同设备可能出现部分设备显示不了图片&#xff0c;解决办法&#xff1a;图片地址直接使用&#xff0c;不要拼接&#xff1a; https://images.weserv.nl/?urlhttp