给定一个字符串 (s) 和一个字符模式 (p)。实现支持 '.' 和 '*' 的正则表达式匹配。

  1. '.' 匹配任意单个字符。
  2. '*' 匹配零个或多个前面的元素。

匹配应该覆盖整个字符串 (s) ,而不是部分字符串。

说明:

  • s 可能为空,且只包含从 a-z 的小写字母。
  • p 可能为空,且只包含从 a-z 的小写字母,以及字符 . 和 *

示例 1:

  1. 输入:
  2. s = "aa"
  3. p = "a"
  4. 输出: false
  5. 解释: "a" 无法匹配 "aa" 整个字符串。

示例 2:

  1. 输入:
  2. s = "aa"
  3. p = "a*"
  4. 输出: true
  5. 解释: '*' 代表可匹配零个或多个前面的元素, 即可以匹配 'a' 。因此, 重复 'a' 一次, 字符串可变为 "aa"

示例 3:

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

示例 4:

  1. 输入:
  2. s = "aab"
  3. p = "c*a*b"
  4. 输出: true
  5. 解释: 'c' 可以不被重复, 'a' 可以被重复一次。因此可以匹配字符串 "aab"

示例 5:

  1. 输入:
  2. s = "mississippi"
  3. p = "mis*is*p*."
  4. 输出: false

思路:如果不用递归的方法,一点一点的判断,比如如果当前的p[j]为*,那么如果s的当前值与p[j-1]相等,就i++。这样存在的问题是,不能判断出*到底代替几个p[j-1],所以我们还是用递归的方法。

  1. bool isMatch(string s, string p) {
  2.  
  3. int pLen=p.length();
  4. int sLen=s.length();
  5.  
  6. if(pLen==0)
  7. return sLen==0;
  8.  
  9. if(pLen==1)
  10. if( (s[0]==p[0] || p[0]=='.') && sLen==1)
  11. return true;
  12. else
  13. return false;
  14.  
  15. if(p[1]!='*')
  16. {
  17. if(s.length()>0 && (s[0]==p[0] || p[0]=='.'))
  18. return isMatch(s.substr(1),p.substr(1));
  19. else
  20. return false;
  21. }
  22. else
  23. {
  24. while(s.length()>0 && (p[0]==s[0] || p[0]=='.'))
  25. {
  26. if(isMatch(s,p.substr(2)))//*表示0个字符
  27. return true;
  28. s=s.substr(1);
  29. }
  30. return isMatch(s,p.substr(2));
  31. }
  32. }
上述程序就将*分成了两种情况,一种是*代表0个字符,一种是*代表1个或多个字符。首先每次判断代表0个字符是否符合,如果不符合,那么就先代表一个,将s的首字符去掉,接着判断,这样*就代表了一个或多个。
还有一种DP的思路,是用空间换时间。额外申请一个数组作为辅助,记录前面的结果。dp[i][j]代表s[0..i-1]和p[0...j-1]的匹配,0代表匹配失败,1代表匹配成功。那么可以得到,dp[0][0]=1表示空串与空串是匹配的,dp[i][0]=0,表示如果p是空串,那么一定不匹配。dp[0][j]表示s是空串,这种情况如果p是x*的组合,是可以匹配的,否则匹配失败。而且dp[0][1]=0表示如果s为空串,p只有一个字符时,肯定匹配失败。dp[i][j]我们可以分为p[j-1]为*和不为*的情况,(1)如果p[j-1]也就是p的最后一个元素为*,那么还可以分为它代表0个前面的元素,也就是说s[0..i-1]与p[0..j-3]是要匹配的,即dp[i][j-2]=1。或者它代表一个或多个前面的元素,那么s[0..i-2]与p[0..j-1]是匹配的,并且s的最后一个元素s[i-1]还要与p的倒数第二个相等,即s[i-1]==p[i-2].
  1. bool isMatch(string s, string p)
  2. {
  3. int m=s.size()+1,n=p.size()+1;
  4. vector<vector<bool> > dp(m,vector<bool>(n));
  5. dp[0][0] = true;//s无字符,p无字符
  6. dp[0][1] = false;//s无字符,p有一个字符(且不能省略)
  7. for (int i = 1; i <= s.size(); ++i)
  8. dp[i][0] = false;//s有字符,p无字符
  9. for (int j = 2; j <= p.size(); ++j)//Ax*只有A与空串匹配,且后两个字符是x*的形式才匹配
  10. dp[0][j] = (p[j - 1] == '*') && dp[0][j - 2];
  11. for (int j = 1; j <= p.size(); ++j)
  12. for (int i = 1; i <= s.size(); ++i) {
  13. if (p[j - 1] != '*') //如果不是*,只有p遍历到.或者p[j-1]和s[i-1]相等的时候匹配
  14. dp[i][j] = dp[i - 1][j - 1] && (p[j - 1] == '.' || s[i - 1] == p[j - 1]);
  15. else dp[i][j] = dp[i][j - 2] ||//x*表示0个x的情况
  16. (dp[i - 1][j - 2] && (p[j - 2] == '.' || p[j - 2] == s[i - 1])) ||//x*表示1个x的情况
  17. //x*表示多个x的时候表示,此时必须s[0~i-2]与p[0~j-1]匹配且……
  18. (dp[i - 1][j] && (p[j - 2] == '.' || p[j - 2] == s[i - 1]));
  19. }
  20. return dp[s.size()][p.size()];
  21. }

Leetcode(10)-正则表达式匹配的更多相关文章

  1. Leetcode 10. 正则表达式匹配 - 题解

    版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C#版 - L ...

  2. Java实现 LeetCode 10 正则表达式匹配

    10. 正则表达式匹配 给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符 '*' 匹配零个或多个前面的那一个元素 所谓匹配, ...

  3. [LeetCode] 10. 正则表达式匹配

    题目链接:https://leetcode-cn.com/problems/regular-expression-matching/ 题目描述: 给定一个字符串 (s) 和一个字符模式 (p).实现支 ...

  4. LeetCode 10. 正则表达式匹配(Regular Expression Matching)

    题目描述 给定一个字符串 (s) 和一个字符模式 (p).实现支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符. '*' 匹配零个或多个前面的元素. 匹配应该覆盖整个字符串 (s ...

  5. LeetCode 10——正则表达式匹配

    1. 题目 2. 解答 在 回溯算法 中我们介绍了一种递归的思路来求解这个问题. 此外,这个问题也可以用动态规划的思路来解决.我们定义状态 \(P[i][j]\) 为子串 \(s[0, i)\) 和 ...

  6. leetcode题目10.正则表达式匹配(困难)

    题目描述: 给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符'*' 匹配零个或多个前面的那一个元素所谓匹配,是要涵盖 整个  ...

  7. 【LeetCode】正则表达式匹配(动态规划)

    题目描述 给定一个字符串 (s) 和一个字符模式 (p).实现支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符. '*' 匹配零个或多个前面的元素. 匹配应该覆盖整个字符串 (s ...

  8. LeetCode10. 正则表达式匹配

    10. 正则表达式匹配 描述 给定一个字符串 (s) 和一个字符模式 (p).实现支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符. '*' 匹配零个或多个前面的元素. 匹配应该 ...

  9. leetcode 10 Regular Expression Matching(简单正则表达式匹配)

    最近代码写的少了,而leetcode一直想做一个python,c/c++解题报告的专题,c/c++一直是我非常喜欢的,c语言编程练习的重要性体现在linux内核编程以及一些大公司算法上机的要求,pyt ...

  10. Leetcode(10)正则表达式匹配

    Leetcode(10)正则表达式匹配 [题目表述]: 给定一个字符串 (s) 和一个字符模式 (p).实现支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符. '*' 匹配零个或 ...

随机推荐

  1. 使用idea插件识别log文件的相关设置

    最近要读一些spring boot项目产生的log文件,众所周知,idea拥有强大的插件系统.当我打开log文件时,idea自动帮我推荐了ideolog这个插件. 但是当我安装好之后发现系统并不能完全 ...

  2. Java程序入门

    编写Java源程序 在d:\day01 目录下新建文本文件,完整的文件名修改为HelloWorld.java ,其中文件名为HelloWorld ,后缀名必须为.java . 用记事本打开 在文件中键 ...

  3. USB过压保护芯片,高输入电压充电器(OVP)

    PW2606B是一种前端过电压和过电流保护装置.它实现了广泛的输入电压范围从2.5VDC到40VDC.过电压阈值可在外部或外部编程设置为内部默认设置.集成功率路径nFET开关的低电阻确保了更好的性能电 ...

  4. Java 8中字符串拼接新姿势:StringJoiner

    介绍 StringJoiner是java.util包中的一个类,用于构造一个由分隔符分隔的字符序列(可选),并且可以从提供的前缀开始并以提供的后缀结尾.虽然这也可以在StringBuilder类的帮助 ...

  5. (04)-Python3之--字典(dict)操作

    1.定义 字典的关键字:dict 字典由多个键和其对应的值构成的 键-值 对组成,每个键值对用冒号 : 分割,每个键值对之间用逗号 , 分割,整个字典包括在花括号 {} 中. {key1:value1 ...

  6. MySQL调优之分区表

    一.分区表的应用场景 1.为什么是用分区表? 表非常大以至于无法全部都放在内存中,或者只在表的最后部分有热点数据,其他均是历史数据,分区表是指根据一定规则,将数据库中的一张表分解成多个更小的,容易管理 ...

  7. java native:Java本地方法调用(jni方式)

    https://www.cnblogs.com/zh1164/p/6283831.html

  8. Python基础(if语句、运算符)

    if语句的简单用法 每条if 语句的核心都是一个值为True 或False 的表达式 (简称条件测试),python根据条件测试的值来判断是否执行if语句后面的代码块,如果为true,执行:为fals ...

  9. Map转换为格式化的YAML字符串

    yaml与java对象的互转 yaml与java对象的互转有snakeyaml <dependency> <groupId>org.yaml</groupId> & ...

  10. 根据table随时添加列

    var newRow = '<tr align="center" class="tdbg" id="tr'+temp[0]+'"> ...