10. 正则表达式匹配

257 阅读2分钟

题目描述

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

  • '.' 匹配任意单个字符
  • '*' 匹配零个或多个前面的那一个元素 所谓匹配,是要涵盖 整个 字符串 s的,而不是部分字符串。

示例

示例 1:
输入:s = "aa" p = "a"
输出:false
解释:"a" 无法匹配 "aa" 整个字符串。

示例 2:
输入:s = "aa" p = "a*"
输出:true
解释:因为 '*' 代表可以匹配零个或多个前面的那一个元素, 在这里前面的元素就是 'a'。因此,字符串 "aa" 可被视为 'a' 重复了一次。

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

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

示例 5:
输入:s = "mississippi" p = "misisp*."
输出:false

来源:力扣(LeetCode)
链接:leetcode-cn.com/problems/re…

实现

bool isMatch(char * s, char * p)
{
    int sLen = strlen(s);
    int pLen = strlen(p);
    bool dp[sLen+1][pLen+1];     // 动态规划数组,s的前i个字符和p的前j个字符是否匹配

    dp[0][0] = true;   // s,p都为空时,可以匹配,因此dp[0][0]=true;
    for (int i = 1; i <= sLen; i++) {
        dp[i][0] = false;        // s不为空,p为空时,无法匹配,dp[i][0]=false;
    }
    for (int j = 1; j <= pLen; j++) {  // s为空,p不为空,这种情况下只有形如:x*y*z*...的形式才能匹配(此时*为0个)
        if (j >= 2 && p[j-1] == '*') {
            dp[0][j] = dp[0][j-2];
        } else {
            dp[0][j] = false;
        }
    }

    // 从上到下,从做到右遍历
    for (int i = 1; i <= sLen; i++) {
        for (int j = 1; j <= pLen; j++) {
            if ((p[j-1] == '.') || (s[i-1] == p[j-1])) {
                dp[i][j] = dp[i-1][j-1];    // 匹配,和上一次的状态相同
            } else if ((j >= 2) && p[j-1] == '*') {
                // *匹配多少个字符不确定
                if (p[j-2] == '.' || s[i-1] == p[j-2]) {
                    /* 1、匹配0个前面字符  dp[i][j] = dp[i][j - 2] 把前面一个字符消掉;
                     * 2、匹配1个前面字符  dp[i][j] = dp[i][j - 1] 相当于没有字符'*';
                     * 3、匹配2个前面字符  dp[i][j] = dp[i - 1][j - 1] 相当于一个前面字符;
                     * 4、匹配多个前面字符 dp[i][j] = dp[i - 1][j] 相当于多个前面字符。
                     */
                    dp[i][j] = dp[i][j-2] || dp[i][j-1] || dp[i-1][j-1] || dp[i-1][j];
                } else {
                    dp[i][j] = dp[i][j-2];    // 不匹配
                }
            } else {
                dp[i][j] = 0;
            }
        }
    }

    return dp[sLen][pLen];  
}

/*
int main(int argc, const char *argv[])
{
    char s[] = "aa";
    char p[] = "a";
    printf("%d\n", isMatch(s, p));
    return 0;
}
*/