刷题记录 回溯算法-10:93. 复原 IP 地址

  题目:93. 复原 IP 地址

有效 IP 地址 正好由四个整数(每个整数位于 0 到 255 之间组成,且不能含有前导 0),整数之间用 '.' 分隔。

  • 例如:"0.1.2.201" 和 "192.168.1.1" 是 有效 IP 地址,但是 "0.011.255.245""192.168.1.312" 和 "192.168@1.1" 是 无效 IP 地址。

给定一个只包含数字的字符串 s ,用以表示一个 IP 地址,返回所有可能的有效 IP 地址,这些地址可以通过在 s 中插入 '.' 来形成。你 不能 重新排序或删除 s 中的任何数字。你可以按 任何 顺序返回答案。

示例 1:

输入:s = "25525511135"
输出:["255.255.11.135","255.255.111.35"]

示例 2:

输入:s = "0000"
输出:["0.0.0.0"]

示例 3:

输入:s = "101023"
输出:["1.0.10.23","1.0.102.3","10.1.0.23","10.10.2.3","101.0.2.3"]

提示:

  • 1 <= s.length <= 20
  • s 仅由数字组成

一、模式识别

1.问题分解

该问题即用回溯法遍历出所有的切割方式,

并逐个判断子串是否合法

因此该题可以分解为切割方式子串合法性判断两个子问题

2.切割问题

从搜索集看为等效的无重复选取无重复元素组合问题

从结果收集条件看为长度条件位置(终点)条件共同作用

所以有双重验证条件:

if len(path) == 4:
            if start_idx == len(s):
                ans.append('.'.join(path))
            return

但本题无需长度剪枝

因为本题中合法的子串对长度也有要求

长度剪枝的作用会被子串合法性判断剪枝的作用覆盖

3.子串合法性判断

总结下来共三个规则:

①以0为开头的子串不合法

②有非正整数字符不合法

③若子串表示的数字大于255不合法

每切割一个子串都需要验证子串的合法性:

(1)验证数字大小 and 对比数字位数与字符位数:

def is_validIP(self, s):
        n = len(s)
        num = int(s)
        if num < 0  or num > 255:
            return False
        t = 0 if not num == 0 else 1
        while num:
            num //= 10
            t += 1
        if n > t:
            return False
        return True

 其中对比数位的操作同时验证①和②两个规则

(2)逐位验证数字 and 验证数字大小:

def is_validIP(self, s):
        n = len(s)
        if s[0] == '0' and len(s) > 1:
            return False
        num = 0
        for i in range(n):
            if not s[i].isdigit():
                return False
            num = num * 10 + int(s[i])
            if num > 255:
                return False
        return True

 

由于子串表示数字大小限制,

因此节点搜索集宽度上限为3:

for i in range(start_idx, min(start_idx + 3, len(s))):

此外子串合法性判断还有个规律:

包含不合法子串的其他字符串也不合法

因此有:

for i in range(start_idx, len(s)):
            ch = s[start_idx: i + 1]
            if self.is_validIP(ch):
                path.append(ch)
                self.backtracking(s, i + 1, path, ans)
                path.pop()
            else:
                break

这条规律与子串长度限制的作用相互覆盖,

选择其一即实现剪枝

二.代码实现

本题中存在字符串,

因此有传递数组拼接传递字符记录点数两种写法

写法一(传递数组拼接):

class Solution:
    def is_validIP(self, s):
        n = len(s)
        num = int(s)
        if num < 0  or num > 255:
            return False
        t = 0 if not num == 0 else 1
        while num:
            num //= 10
            t += 1
        if n > t:
            return False
        return True

    def backtracking(self, s, start_idx, path, ans):
        if len(path) == 4:
            if start_idx == len(s):
                ans.append('.'.join(path))
            return
        
        for i in range(start_idx, len(s)):
            ch = s[start_idx: i + 1]
            if self.is_validIP(ch):
                path.append(ch)
                self.backtracking(s, i + 1, path, ans)
                path.pop()
            else:
                break

    def restoreIpAddresses(self, s: str) -> List[str]:
        ans = []
        self.backtracking(s, 0, [], ans)
        return ans

写法二(传递字符串记录点数): 

class Solution:
    def is_valid(self, s):
        n = len(s)
        if n == 0:
            return False
        num = 0
        if n > 1 and s[0] == '0':
            return False
        for i in range(n):
            if not s[i].isdigit():
                return False
            num = num * 10 + int(s[i])
        return 0 <= num <= 255

    def backtracking(self, s, start_idx, pointnum, path, ans):
        if pointnum == 3:
            if self.is_valid(s[start_idx:]):
                ans.append(path + s[start_idx:])
            return
        
        for i in range(start_idx, len(s)):
            ch = s[start_idx: i + 1]
            if self.is_valid(ch):
                self.backtracking(s, i + 1, pointnum + 1, path + ch + '.', ans)
            else:
                break

    def restoreIpAddresses(self, s: str) -> List[str]:
        ans = []
        self.backtracking(s, 0, 0, "", ans)
        return ans

此外,验证函数的逻辑可以写在for循环内部 

写法三(for循环内部验证合法性): 

class Solution:
    def backtracking(self, s, start_idx, path, ans):
        if len(path) == 4:
            if start_idx == len(s):
                ans.append('.'.join(path))
            return
        
        for i in range(start_idx, len(s)):
            if s[start_idx] == '0' and i > start_idx:
                break
            # 长度决定的两种不可能完成路径的情况:
            # 剩余路径的最大容量少于剩余的数字:3 * (4 - len(path)) < (len(s) - i)
            # 剩余路径的最小需求多于剩余的数字:(4 - len(path)) > (len(s) - i)
            if 3 * (4 - len(path)) < (len(s) - i) or (4 - len(path)) > (len(s) - i):
                break
            ch = s[start_idx: i + 1]
            if int(ch) > 255:
                break
            path.append(ch)
            self.backtracking(s, i + 1, path, ans)
            path.pop()

    def restoreIpAddresses(self, s: str) -> List[str]:
        ans = []
        self.backtracking(s, 0, [], ans)
        return ans

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

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

相关文章

【大数据】机器学习------神经网络模型

一、神经网络模型 1. 基本概念 神经网络是一种模拟人类大脑神经元结构的计算模型&#xff0c;由多个神经元&#xff08;节点&#xff09;组成&#xff0c;这些节点按照不同层次排列&#xff0c;通常包括输入层、一个或多个隐藏层和输出层。每个神经元接收来自上一层神经元的输…

docker一张图理解

1、push 将本地的镜像上传到镜像仓库,要先登陆到镜像仓库。参数说明&#xff1a; –disable-content-trust : 忽略镜像的校验,默认开启 # 上传本地镜像myapache:v1到镜像仓库中。 docker push myapache:v1 1.2、search 从Docker Hub查找镜像。参数说明&#xff1a; –…

Unity shader中真的可以动态关闭Stencil Test吗?

这个问题很多年前就有人问了&#xff1a; https://discussions.unity.com/t/how-to-disable-the-stencil-block-via-shader-properties/600273/1 最后的答案是&#xff1a; set [_StencilComp] to CompareFunction.Disabled to disable the Stencil Op completely. 但是我测试…

Python----Python高级(函数基础,形参和实参,参数传递,全局变量和局部变量,匿名函数,递归函数,eval()函数,LEGB规则)

一、函数基础 1.1、函数的用法和底层分析 函数是可重用的程序代码块。 函数的作用&#xff0c;不仅可以实现代码的复用&#xff0c;更能实现代码的一致性。一致性指的是&#xff0c;只要修改函数的代码&#xff0c;则所有调用该函数的地方都能得到体现。 在编写函数时&#xf…

win10电脑 定时关机

win10电脑 定时关机 https://weibo.com/ttarticle/p/show?id2309405110707766296723 二、使用任务计划程序设置定时关机打开任务计划程序&#xff1a; 按下“Win S”组合键&#xff0c;打开搜索框。 在搜索框中输入“任务计划程序”&#xff0c;然后点击搜索结果中的“任务…

初识JAVA-面向对象的三大特征之多态

1. 重温面向对象 面向对象是一种解决问题的思想&#xff0c;它把计算机程序看作是各种对象组合起来的。每个对象都有自己的数据&#xff08;属性&#xff09;和行为&#xff08;方法&#xff09;&#xff0c;主要依靠对象之间的交互来解决和实现问题。Java是一门纯面向对象的语…

2024年11月架构设计师综合知识真题回顾,附参考答案、解析及所涉知识点(一)

软考高级系统架构设计师考试包含三个科目&#xff1a;信息系统综合知识、系统架构设计案例分析和系统架构设计论文。考试形式为机考。本文主要回顾2024年下半年(2024-11-10)系统架构设计师考试上午综合知识科目的选择题&#xff0c;同时附带参考答案、解析和所涉知识点。 由于机…

【STM32-学习笔记-8-】I2C通信

文章目录 I2C通信Ⅰ、硬件电路Ⅱ、IIC时序基本单元① 起始条件② 终止条件③ 发送一个字节④ 接收一个字节⑤ 发送应答⑥ 接收应答 Ⅲ、IIC时序① 指定地址写② 当前地址读③ 指定地址读 Ⅳ、MPU6050---6轴姿态传感器&#xff08;软件I2C&#xff09;1、模块内部电路2、寄存器地…

Angular-生命周期及钩子函数

什么是生命周期 Angular 创建和渲染组件及其子组件&#xff0c;当它们绑定的属性发生变化时检查它们&#xff0c;并在从 DOM 中移除它之前销毁它们。生命周期函数通俗的讲就是组件创建、组件更新、组件销毁的时候会触发的一系列的方法。当 Angular 使用构造函数新建一个组件或…

【计算机网络】深入浅出计算机网络

第一章 计算机网络在信息时代的作用 计算机网络已由一种通信基础设施发展成一种重要的信息服务基础设施 CNNIC 中国互联网网络信息中心 因特网概述 网络、互联网和因特网 网络&#xff08;Network&#xff09;由若干结点&#xff08;Node&#xff09;和连接这些结点的链路…

Golang——rune和byte

本文详细介绍Golang中的两种字符类型rune和byte&#xff0c;介绍他们的区别&#xff0c;编码方式和简单的使用。 文章目录 byte 类型rune 类型UTF-8 与 Unicode 的关系byte和rune的主要区别Go的默认编码方式遍历方式遍历 byte遍历 rune补充 字符还原从 byte 序列还原字符串从 r…

基于当前最前沿的前端(Vue3 + Vite + Antdv)和后台(Spring boot)实现的低代码开发平台

项目是一个基于当前最前沿的前端技术栈&#xff08;Vue3 Vite Ant Design Vue&#xff0c;简称Antdv&#xff09;和后台技术栈&#xff08;Spring Boot&#xff09;实现的低代码开发平台。以下是对该项目的详细介绍&#xff1a; 一、项目概述 项目名称&#xff1a;lowcode-s…

java springboot3.x jwt+spring security6.x实现用户登录认证

springboot3.x jwtspring security6.x实现用户登录认证 什么是JWT JWT&#xff08;JSON Web Token&#xff09;是一种开放标准&#xff08;RFC 7519&#xff09;&#xff0c;它用于在网络应用环境中传递声明。通常&#xff0c;JWT用于身份验证和信息交换。JWT的一个典型用法是…

代码随想录刷题day07|(数组篇)58.区间和

目录 一、数组理论基础 二、前缀和 三、相关算法题目 四、总结 五、待解决问题 一、数组理论基础 数组是存放在连续内存空间上的相同类型数据的集合。 代码随想录 (programmercarl.com) 特点&#xff1a; 1.下标从0开始&#xff0c;内存中地址空间是连续的 2.查询快&…

专用小软件,完全免费,非常丝滑

今天给大家介绍一个专门将PDF数电发票合并打印的软件&#xff0c;这个软件可以批量操作&#xff0c;完全免费没有任何的广告。 电子发票专用批量打印工具 免费批量使用 软件无需安装&#xff0c;解压之后双击这个图标就能直接使用了。 点击右上角的加号&#xff0c;选中需要打…

安装虚拟机VMware遇到的问题

问题1&#xff1a;进入如下界面&#xff0c;不知道如何操作 解决办法 键盘⬇️&#xff0c;选择“Reset the system”回车 问题2&#xff1a;系统存放位置我给放在了VMware安装目录&#xff0c;具体D:\software\VMware\Windows安装不行 解决办法&#xff1a;D:\software\virt…

Matlab 具有周期性分布的死角孔的饱和空气多孔材料的声学特性

本文对直主孔含侧空腔&#xff08;死角&#xff09;的饱和空气多孔介质中的声传播进行了理论和数值研究。侧腔位于沿每个主孔周期性间隔的“节点”上。研究了侧向空腔分布中周期性的影响&#xff0c;并单独考虑了紧间隔死角的低频极限。结果表明&#xff0c;吸附系数和透射损失…

Vue如何构建项目

目录 1.安装Node.js 2.换源(建议) 3.选择一个目录 4.创建一个vue项目 5.验证是否成功 1.安装Node.js 安装18.3或更⾼版本的 Nodejs 点击下载->Node.Js中文网 node -v npm -v 安装好后在windows的cmd窗口下运行 如果能运行出结果就说明安装好了。 2.换源(建议) //…

网络层协议-----IP协议

目录 1.认识IP地址 2.IP地址的分类 3.子网划分 4.公网IP和私网IP 5.IP协议 6.如何解决IP地址不够用 1.认识IP地址 IP 地址&#xff08;Internet Protocol Address&#xff09;是指互联网协议地址。 它是分配给连接到互联网的设备&#xff08;如计算机、服务器、智能手机…

MacOS 下 Memory Analyzer 启动报错

&#x1f468;&#x1f3fb;‍&#x1f4bb; 热爱摄影的程序员 &#x1f468;&#x1f3fb;‍&#x1f3a8; 喜欢编码的设计师 &#x1f9d5;&#x1f3fb; 擅长设计的剪辑师 &#x1f9d1;&#x1f3fb;‍&#x1f3eb; 一位高冷无情的全栈工程师 欢迎分享 / 收藏 / 赞 / 在看…