当前位置: 首页 > article >正文

剑指 Offer(第2版)面试题 19:正则表达式匹配

剑指 Offer(第2版)面试题 19:正则表达式匹配

  • 剑指 Offer(第2版)面试题 19:正则表达式匹配
    • 解法1:递归
    • 解法2:动态规划

剑指 Offer(第2版)面试题 19:正则表达式匹配

题目来源:

  1. AcWing 30. 正则表达式匹配
  2. LeetCode 10. 正则表达式匹配

解法1:递归

代码:

class Solution
{
public:
    bool isMatch(string s, string p)
    {
        if (p.empty())
            return s.empty();
        return match(s, p, 0, 0);
    }
    bool match(string s, string p, int a, int b)
    {
        if (b == p.size())
            return a == s.size();
        else if (a == s.size())
        {
            if (b == p.size() - 1 || p[b + 1] != '*')
                return false;
            return match(s, p, a, b + 2);
        }
        if (b + 1 == p.size())
            return a + 1 == s.size() && (s[a] == p[b] || p[b] == '.');
        if (p[b + 1] != '*')
        {
            if (s[a] == p[b] || p[b] == '.')
                return match(s, p, a + 1, b + 1);
            return false;
        }
        if (s[a] != p[b] && p[b] != '.')
            return match(s, p, a, b + 2);
        return match(s, p, a, b + 2) || match(s, p, a + 1, b + 2) || match(s, p, a + 1, b);
    }
};

注:这种写法不能在 LeetCode 上通过,爆栈了。

解法2:动态规划

代码:

class Solution
{
public:
    bool isMatch(string s, string p)
    {
        int m = s.size(), n = p.size();
        // 状态矩阵
        vector<vector<bool>> dp(m + 1, vector<bool>(n + 1, false));
        // dp[i,j]: 表示以i截止的字符串是否可以被以j截止的正则表达式匹配
        //  初始化
        dp[0][0] = true;
        // 在匹配0次的情况下,浪费一个字符+星号的组合,没有匹配任何s中的字符
        for (int i = 1; i <= n; i++)
            if (p[i - 1] == '*') //'*'匹配零个或多个前面的那一个元素
                dp[0][i] = dp[0][i - 2];
        // 状态转移
        for (int i = 1; i <= m; i++)
            for (int j = 1; j <= n; j++)
            {
                if (p[j - 1] != '*') // 当前p匹配的字符不是'*'
                {
                    if (match(s[i - 1], p[j - 1]) == true) // 当前s匹配的字符和p匹配的字符相符合
                        dp[i][j] = dp[i - 1][j - 1];
                    else // 当前s匹配的字符和p匹配的字符不相符合,dp[i,j]=false
                        dp[i][j] = false;
                }
                else // 当前p匹配的字符为'*'
                {
                    // 当前s匹配的字符和p匹配的p中一个字符+星号的组合相匹配,
                    // 要么跳过一个字符+星号的组合,要么看dp[i-1][j]是否为true,或运算连接
                    if (match(s[i - 1], p[j - 2]) == true)
                        dp[i][j] = dp[i - 1][j] || dp[i][j - 2];
                    else // 当前s匹配的字符和p中一个字符+星号的组合不相匹配,跳过该组合
                        dp[i][j] = dp[i][j - 2];
                }
            }
        return dp[m][n];
    }
    // 辅函数
    bool match(char &sc, char &pc)
    {
        //'.'匹配任意单个字符,故一定匹配成功
        if (pc == '.')
            return true;
        // 不然看s字符和p字符是否相同
        return sc == pc;
    }
};

复杂度分析:

时间复杂度:O(m*n),其中 m 是字符串 s 的长度,n 是字符串 p 的长度。

空间复杂度:O(m*n),其中 m 是字符串 s 的长度,n 是字符串 p 的长度。


http://www.kler.cn/a/160862.html

相关文章:

  • 【GESP】C++二级练习 luogu-B2079, 求出 e 的值
  • WPF中组件之间传递参数的方法研究
  • 在Windows环境下搭建无人机模拟器
  • Linux web服务器
  • 本地手集博客id“升级”在线抓取——简陋版——(2024年终总结1.1)
  • 【llm/ollama/qwen】在本地部署qwen2.5-coder并在vscode中集成使用代码提示功能
  • Linux中的日志管理
  • 【智能家居】三、添加语音识别模块的串口读取功能点
  • Java语言中的修饰符
  • 统计centos系统哪一个进程打开文件描述符
  • CSS 在性能优化方面的实践
  • 使用pytorch从零开始实现迷你GPT
  • R语言手册30分钟上手
  • Javase | 贪吃蛇小游戏
  • L1-025:正整数A+B
  • python筛选并删除两个文件夹中不同文件名的文件
  • 2-redis高级-centos上安装redis(编译安装、redis启动)、redis客户端操作、redis使用场景、redis中的通用命令
  • Google Guava 集合工具使用详解
  • Node.js快速搭建简单的HTTP服务器并发布公网远程访问
  • 多向通信----多人聊天
  • Elaticsearch 学习笔记
  • 量子纠缠通讯:未来信息技术的革新者
  • AI算力研究报告:智算供给格局分化国产化进程有望加速
  • 【华为网络-配置-025】- 同 VLAN 下不同网段通信(启用 Sub 地址)
  • 圆通单号查询,圆通速递物流查询,对需要的单号进行颜色标记
  • 你了解架构图吗?