10. 正则表达式匹配

给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 '.' 和 '*' 的正则表达式匹配。

  • '.' 匹配任意单个字符
  • '*' 匹配零个或多个前面的那一个元素

所谓匹配,是要涵盖 整个 字符串 s的,而不是部分字符串。

示例 1:
输入:s = "ab" p = "."
输出:true
解释:".
" 表示可匹配零个或多个('*')任意字符('.')。

示例 2:
输入:s = "aab" p = "cab"
输出:true
解释:因为 '*' 表示零个或多个,这里 'c' 为 0 个, 'a' 被重复一次。因此可以匹配字符串 "aab"。

解法

这个题确实比较难,自己昨天没看答案,花了好几个小时也没有做出来,总有最后几个通过不了测试。也想过用动态规划来做,但是不知道怎么推出动态转移方程,所以采用了双指针的做法,但是没有做出来。

力扣的官方答案不太好理解,最后还是参考了力扣的一个精选视频,才完全搞懂,以后碰到不好推出动态转移方程的,可以直接弄一张表,自己先填写一下,填着填着就能发现规律了。

填写完成过后的截图

代码如下,附有详细的注释,结合着这张图看代码一定会瞬间明白的:

def isMatch2(s, p):
    m, n = len(s), len(p)
    # 初始化二维数组,以 " "+p 为行的头,以" "+s为列的头
    # 如果s和p完全匹配,那么它们两个头部加上一个" "空字符也同样匹配
    # 之所以加一个空字符,是为了方便初始化,即第一行即为初始条件
    dp = [[False] * (n + 1) for _ in range(m + 1)]
    dp[0][0] = True  # 最左上角,因为都是两个空串,所以匹配,为True

    # 初始化p与空串的匹配,即先填写第一行,初始条件
    for col in range(1, n + 1):
        # 即从p的第二位开始
        if col > 1:
            # 如果当前为*号,那么它的状态值就等于左边第二位的状态值
            # 因为前面加了一个空字符,所以p的当前字符下标都要减1,s与p相同
            if p[col - 1] == "*":
                dp[0][col] = dp[0][col - 2]
            else:
                dp[0][col] = False
        else:
            # 这里其实是col等于1的时候,也就是p的第一位字符
            # 因为p不会以"*"或者" "开头,所以肯定不会走到这个else里面来
            # dp[0][1]肯定等于False,这个else可以省略
            if p[col - 1] == "*":
                dp[0][col] = True

    # 开始填表,从dp[1][1]开始每行每行的开始填表
    for row in range(1, m + 1):
        for col in range(1, n + 1):
            # 如果p的当前字符为"."或者等于s的当前字符
            # 那么它的状态值就取决于它左上角的状态值
            # 而且p的当前字符肯定不为"*"
            if p[col - 1] == s[row - 1] or p[col - 1] == ".":
                dp[row][col] = dp[row - 1][col - 1]
            elif p[col - 1] == "*":  # p的当前字符为"*"的情况下
                # 因为"*"不可能出现在p的第一位,所以其实下面的col>1的判断可以省略
                if col > 1:
                    # 当p的当前位为"*"号时,如果p的当前左边第二位也和s的当前位匹配,那么p的当前位也匹配
                    # 也就是 "*"前面字符出现0次的情况
                    if dp[row][col - 2]:
                        dp[row][col] = True
                    # 如果p当前的前一位字符为"."或者等于s的当前字符
                    # 那么当前位置的状态就等于它上面第一个位置的状态
                    # 这也是 "*"前面字符出现多次的情况
                    elif p[col - 2] in {s[row - 1], "."}:
                        dp[row][col] = dp[row - 1][col]

    # 返回填表后最右下角的值,即为整个p和s的匹配情况
    return dp[m][n]
复杂度分析
  • 时间复杂度:O(mn),其中 m 和 n 分别是字符串 s 和 p 的长度。我们需要计算出所有的状态,并且每个状态在进行转移时的时间复杂度为O(1)。因为我们是从dp[1][1]开始填的,所以时间复杂度还是O(mn)。
  • 空间复杂度:O((m+1)(n+1)),即为存储所有状态使用的空间。因为s和p前面都加了一个空字符,所以空间复杂度是O((m+1)(n+1))。

力扣官方答案

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容