正则表达式匹配——力扣困难题解

news2024/11/15 11:41:21

力扣链接:正则表达式匹配
题目描述
给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 ‘.’ 和 ‘*’ 的正则表达式匹配。

‘.’ 匹配任意单个字符
‘*’ 匹配零个或多个前面的那一个元素
所谓匹配,是要涵盖 整个 字符串 s 的,而不是部分字符串。
在这里插入图片描述
在这里插入图片描述

解题思路

分析字符串

根据上面的示例,首先我们知道 s s s只包含小写字母; p p p包含小写字母、‘.’ 和 ‘*’。
先考虑比较简单的情况,即 p p p不包含’*'。

简单情况的判别

思路如下:

  1. 判断 s s s p p p的长度是否一致。不一致的话,直接return False;一致的话,接着第二步。
  2. 遍历访问 s s s p p p对应位置上的元素,判断是否相等。相等的话(包括 p [ i ] =   ′ .   ′ p[i] = \ '.\ ' p[i]= . ),继续比对;不相等,直接return False。
for i in range(n):
	if s[i]==p[i] or p[i] == '.':
		continue
	else:
		return False
复杂情况的判别——p含有’*’
  1. s = a a b s=aab s=aab, p = a a b c ∗ p=aabc* p=aabc
    p中有 ∗ * ∗ * 前面是c,所以该位置,c可以出现0次、1次或多次
    该case的返回应该是True,即c出现0次 p = a a b p=aab p=aab -> s = p s=p s=p
  2. s = a a c d b f s=aacdbf s=aacdbf, p = a a c d b ∗ f p=aacdb*f p=aacdbf
    p中有 ∗ * ∗ * 前面是b,所以该位置,b可以出现0次、1次或多次
    该case的返回应该是True,即b出现1次 p = a a c d b f p=aacdbf p=aacdbf -> s = p s=p s=p
  3. s = a a c d b b f s=aacdbbf s=aacdbbf, p = a a c d b ∗ f p=aacdb*f p=aacdbf
    p中有 ∗ * ∗ * 前面是b,所以该位置,b可以出现0次、1次或多次
    该case的返回应该是True,即b出现多次 p = a a c d b b f p=aacdbbf p=aacdbbf -> s = p s=p s=p

算法——动态规划

当p含有多个 ∗ * 时,我们无法判别每次 ∗ * 出现时,它前面的那个字符需要出现几次,这种属于动态规划的范围,即都是可商量的。

动态规划——边界情况处理

  1. 定义:
    f [ i ] [ j ] f[i][j] f[i][j]:s[0: i-1]和p[0: j-1]的匹配情况。
    i,j代表的是长度(因为动态规划有个初始值,为了定义初始值一般会有这种错位的情况,即不能认为i,j是下标),所以对应下标就是i-1,j-1。
  2. 初始化数组:
    f=[[False]*(n+1) for _ in range(m+1)]
    初始化边界情况:
    i=0,j=0,f[0][0] = True;
    i=0,j>0,存在f[0][j]= True的情况,需要处理
    i>0,j=0,不存在f[i][0]= True的情况,无需处理
		m=len(s)
        n=len(p)
        f=[[False]*(n+1) for _ in range(m+1)]
        # 初始化f[0][0]
        f[0][0] = True
        # 初始化, s:没有字符,p有字符的匹配情况
        for j in range(1,n+1):
            if p[j-1] == '*': 
            # 只能匹配0次,查看是否有True的可能
            #【0次】匹配才能满足【p的变化结果】没有字符
                f[0][j] = f[0][j-2] # f[0][-1]=f[0][n],不用担心越界

s = a a b s=aab s=aab, p = a a b c ∗ p=aabc* p=aabc,

f[0][0]=True
# 因为 f=[[False]*(n+1) for _ in range(m+1)]
f[0][1]=False 
f[0][2]=False
f[0][3]=False
f[0][4]=False
# 可能True的,会进行动态规划转移
f[0][5]=f[0][3]=False

如果还没看明白,为什么遇到*号,有状态转移,请看下一个例子
s = a a b s=aab s=aab, p = a ∗ a ∗ b ∗ c ∗ p=a*a*b*c* p=aabc,

f[0][0]=True
# 不满足if条件的默认是False,因为 f=[[False]*(n+1) for _ in range(m+1)]
# 可能True的,会进行动态规划转移
f[0][1]=False 
f[0][2]=f[0][0]=True
f[0][3]=False
f[0][4]=f[0][2]=True
f[0][5]=False
f[0][6]=f[0][4]=True
f[0][7]=True
f[0][8]=f[0][6]=True

动态规划——状态转移

p[j-1]不是 ∗ * 时,正常匹配。

if p[j-1]=='.' or s[i-1]==p[j-1]: # 该位置匹配上了,f[i][j]的情况由f[i-1][j-1]决定
   f[i][j] = f[i-1][j-1] 

p[j-1]是 ∗ * :
如果 ∗ * 号前的字符和s[i-1]匹配上了,可能0次、1次或多次;
如果 ∗ * 号前的字符p[j-2]和s[i-1]没匹配上,只能0次,即舍弃p[j-2]、p[j-1]; (不用担心p[0]=‘*’,因为题目保证了*号前面有有效字符)

if s[i-1] != p[j-2] and p[j-2]!='.': # s[i-1],p[j-2]没匹配上,只能0次
     f[i][j]=f[i][j-2]
else: # 匹配上了,尝试0次、1次或多次,三者有一个满足True, f[i][j]就是True
     f[i][j] = f[i][j-2] | f[i-1][j]

0次: f[i][j] 转移到 f[i][j-2]
1次或多次: f[i][j] 转移到 f[i-1][j],即删掉s[i-1], f[i][j] 的匹配情况由s[0:i-2]和p[0:j-1]的匹配情况(0次、1次或多次)决定。
比如s=‘aaaa’和p=’a*‘,
f[4][2]=f[3][2] # 由4次,转换为3次, 判断s=‘aaa’和p=’a*’
f[3][2]=f[2][2] # 由3次,转换为2次, 判断s=‘aa’和p=’a*’
f[2][2]=f[1][2] # 由2次,转换为1次, 判断s=‘a’和p=’a*’
f[1][2]=f[0][2] # 由1次,转换为0次, 判断s=‘‘和p=’a*’
f[0][2]=f[0][0] # 0次,0次 , 判断s=‘‘和p=’’

解题代码

class Solution:
    def isMatch(self, s: str, p: str) -> bool:
        m=len(s)
        n=len(p)
        f=[[False]*(n+1) for _ in range(m+1)]
        # 初始化[0][0]
        f[0][0] = True
        # 初始化 '*' f[0][j],匹配0个
        for j in range(1,n+1):
            if p[j-1] == '*':
                f[0][j] = f[0][j-2] # f[0][-1]=f[0][n],不用担心越界

        for i in range(1,m+1):
            for j in range(1,n+1):
                if p[j-1]=='.' or s[i-1]==p[j-1]: # 小写字母匹配上 或 通用的
                    f[i][j] = f[i-1][j-1]
                elif p[j-1] == '*': # 可0次,1次,多次
                    if s[i-1] != p[j-2] and p[j-2]!='.': # s[i-1],p[j-2]没匹配上,只能0次
                        f[i][j]=f[i][j-2]
                    else: # 匹配上了,尝试0次、1次或多次,三者有一个满足True, f[i][j]就是True
                        f[i][j] = f[i][j-2] | f[i-1][j]
        return f[m][n]

复杂度

时间: O ( m n ) O(mn) O(mn),其中 m 和 n 分别是字符串 s 和 p 的长度。我们需要计算出所有的状态,并且每个状态在进行转移时的时间复杂度为 O(1)。
空间: O ( m n ) O(mn) O(mn),即为存储所有状态使用的空间。

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

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

相关文章

GUI界面开发之tkinter(三) 按钮类组件和选择列表类组件

大家好!我是码银儿~,欢迎关注🥰: CSDN:码银公众号:码银学编程 一、按钮类组件 按钮类组件顾名思义就是按钮,跟平时大家看见的按钮没啥区别,允许用户通过点击执行操作。以下是三种…

Awesome-LLMs-for-Video-Understanding - 基于大型语言模型的视频理解研究

Awesome-LLMs-for-Video-Understanding 是 基于大型语言模型的视频理解研究 github : https://github.com/yunlong10/Awesome-LLMs-for-Video-Understandingpaper:Video Understanding with Large Language Models: A Survey https://arxiv.org/pdf/2312.17432 视频…

五、前后端分离通用权限系统(5)

🌻🌻 目录 一、前端框架1.1、vue-element-admin1.1.1、Vue 概述1.1.2、Element-ui 概述1.1.3、ES6 概述 1.2、vue-admin-template1.2.1、简介1.2.2、下载1.2.3、安装1.2.4、源码目录结构(了解)1.2.5、改造登录&退出功能1.2.5.…

跨域解决 | 面试常问问题

跨域解决 | 面试常问问题 跨域问题一直是前端开发中不可避免的一部分,它涉及到浏览器的同源策略和安全机制。本文将深入解析跨域问题的本质,并探讨前端和后端的多种解决方案,同时分享一些扩展与高级技巧。最后,我们还将总结跨域解…

K8S系列——(二)、K8S部署RocketMQ集群

1、环境准备 要将RocketMQ部署到K8S上,首先你需要提前准备一个K8S集群环境,如图我已经准备好了一个版本为 v1.28.13 的 K8S 集群(其他版本也没问题): 角色IPMaster192.168.6.220Node-1192.168.6.221Node-2192.168.6.…

浏览器不开梯子无法上网,检查代理或防火墙或者找不到服务器ip地址

1、代理没有关闭 检查代理是否关闭 检查方法1: 在控制面版中找到Internet选项,点击连接栏,在连接栏中选择局域网设置。之后将代理服务器下面的框选中的对勾取消。最终如下 检查方法2: 打开设置,找到网络和internet…

书生浦语大模型实战营:LMDeploy量化部署

1.任务: 使用结合W4A16量化与kv cache量化的internlm2_5-1_8b-chat模型封装本地API并与大模型进行一次对话。 2.背景: 1.计算模型需要的权重大小: 1B代表10个亿参数,假如是16位浮点数(f16),也…

计算机视觉概念科普

计算机视觉(Computer Vision, CV)是一门多学科交叉的科学,旨在让计算机具备“看”的能力,即通过图像或视频数据来理解世界。它结合了信号处理、图像处理、模式识别、机器学习等多个领域的技术,让计算机能够执行诸如识别…

【Python学习手册(第四版)】学习笔记20.2-迭代和解析(二)-迭代解析、迭代方法的计时比较、函数陷阱

个人总结难免疏漏,请多包涵。更多内容请查看原文。本文以及学习笔记系列仅用于个人学习、研究交流。 本文较简单,主要是概括了解析语法(列表解析、生成器、集合、字典解析),以及对前面的各种迭代进行计时比较&#xf…

通过python解决原神解密

最近楼主玩原神世界任务做到稻妻了,在稻妻有很多解密游戏,但是博主最头疼的就是稻妻的石头解密QAQ(如图) 就在昨晚,楼主又碰到了石头解密,瞎打,半天解不出来。于是就想,有没有什么严…

如何在Windows下使用make编译Makefile

最近有小伙伴咨询我去编译运行一个程序。我一开始以为是CMakeLists,结果发现是makefile。 什么是Makefile ‌Makefile是一种用于自动化构建和管理程序的工具‌,它定义了项目中文件的依赖关系和构建步骤,帮助程序员自动化编译、链接和打包程序…

Ps:创建帧动画

在 Photoshop 中,帧动画 Frame Animation是一种通过在“时间轴”面板中创建和管理多个帧来实现动画效果的方式。 所谓帧动画,也就是传统意义上的逐帧动画,依次播放每个帧而构成的动画形式。每个帧记录了“图层”面板上所有图层的属性状态&…

QT Mainwindow下指定控件的setMouseTracking(true)和mousemoveevent函数失效-问题解决

目录: 一,问题描述二,解决方法2.1解决依据2.2方法实操 三,参考资料 一,问题描述 ☀️之前碰到过的一个问题,现在分享出来:想在qt哪里搞个鼠标移动在控件显示的图片上,然后实时显示对…

[数据集][目标检测]红外场景下车辆和行人检测数据集VOC+YOLO格式19069张4类别

数据集格式:Pascal VOC格式YOLO格式(不包含分割路径的txt文件,仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数):19069 标注数量(xml文件个数):19069 标注数量(txt文件个数):19069 标…

一文带你画PCB板,有手就行

背景 最近写Autosar网络唤醒功能,想在实际硬件上验证,但是市面上没有找到板子验证,只能找人帮忙画PCB板。但是这里遇到比较大的问题,1、整个周期会比较长,板子不太可能一次就能完成,中间会出现修改的地方&…

C#/.NET/.NET Core技术前沿周刊 | 第 1 期(2024年8.12-8.18)

前言 C#/.NET/.NET Core技术前沿周刊,你的每周技术指南针!记录、追踪C#/.NET/.NET Core领域、生态的每周最新、最实用的技术文章、社区动态、优质项目和学习资源等。让你时刻站在技术前沿,助力技术成长与视野拓宽。 欢迎投稿,推荐…

【蓝牙协议栈】【BLE】【GATT】精讲GATT Profile架构(图文并茂精华版)

目录 1. 蓝牙BLE通用属性协议(GATT) 2.GATT角色介绍 3.GATT 层级和Profile架构 4. GATT Server架构(重点内容) 5. Characteristic架构(重点内容) 6. (Characteristic Properties)特性属性详解&#x…

一款免费的目录以及文件对比工具,替代Beyond Compare

Beyond Compare是一款功能强大的文件对比工具,但是由于是付费软件,很多没有购买的企业用户无法使用。CCompare就是一款替代Beyond Compare的免费方案,一款来自中国的可替换beycond compare, 免费使用的代码同步对比工具。 CCompare 是一款功…

【科研绘图】【分条热力图】:附Origin详细画图流程 + 案例分析

目录 No.1 理解分条热力图 No.2 画图流程 1 导入数据,绘制图形 2 设置绘图细节 3 色阶控制 4 设置坐标轴 5 效果图 No.3 案例分析 1 案例一 2 案例二 No.1 理解分条热力图 分条热力图,基于数据映射和颜色编码,是在热力图的基础上进…

聚星文社下载地址

聚星文社绘唐科技是一个文化传媒公司,以绘制唐朝相关的科技设备和场景为主题。该公司致力于通过插画、动画、游戏等形式,栩栩如生地展现唐朝时期的科技发展和生活场景。 聚星文社下载地址https://iimenvrieak.feishu.cn/docx/ZhRNdEWT6oGdCwxdhOPcdds7n…