LeetCode——10. Regular Expression Matching
一.题目链接:https://leetcode.com/problems/regular-expression-matching/
二.题目大意:
实现一个正则表达式,该正则表达式只有两种特殊的字符——“.”和“*”,其中.能表示任意字符,即它可以匹配任意的字符;*表示可以重复前面的字符0次或者多次(例如:a*可以表示成“”(空,相当于重复0次)或者表示成“aaa”重复3次)。
三.题解:
这道题目比较常用的方法是递归;该题目的难点之处在于遇到*的时候它可能匹配0次,也可能匹配1次或多次;这种特性很显然满足递归的特性。所以要根据字符是否为*来进行分情况讨论,详细如下:
对于一个模式字符串,从前往后的来分析它:
1.如果当前字符的下一个字符不是“*”的话(由于第一个字符不可能是* 所以根据下一个字符是否为*来分情况才更加的合理),此时只需要判断需匹配字符串s的当前字符和模式字符串p的当前字符是否相等即可。如果*s == *p或者*p == '.' && *s != '\0'的话,那么当前字符匹配成功,就可以继续往下匹配了。
2.如果当前字符的下一个字符是“*”的话,此时就需要匹配当前字符0次、1次或多次了(而此处的重复次数可以用递归法来实现,也可以用迭代法来实现),直到不能匹配更多的字符。
3.既然是递归,那么一定·要有终止条件的,该问题的终止条件显然是:当p到达终点时(即*p == '\0'时)如果s也到达终点了,那么返回true;如果s没到达终点,那么返回false。通过这个条件我们也可以看出,字符串p的最终长度一定是与字符串s的长度一样,才能够匹配成功。(这也就是题目时所说的完全匹配,而不是部分匹配。通过这一点,我们可也快速的去除那些不符合匹配的例子,如:ab和.*c,两者显然不匹配;而ab和.* 两者显然匹配)
具体代码如下:
方法1(递归+部分迭代):
class Solution {
public:
bool isMatch(string s, string p) {
return matchCore(s.c_str(),p.c_str());
}
bool matchCore(const char *s,const char *p)
{
if(*p == '\0' && *s == '\0')
return true;
if(*p == '\0' && *s != '\0')
return false;
//如果下一个字符不是*的话
if(*(p+1) != '*')
{
if(*p == *s||(*p == '.' && *s != '\0'))
return matchCore(s + 1,p + 1);
else//一旦有一个字符不一样,则不匹配
return false;
}
else//下一个字符是*
{
//*重复一次或多次,只有当当前字符匹配成功时,才考虑*重复多少次,否则不用考虑*了,直接跳过
while(*p == *s ||(*p == '.' && *s != '\0'))
{
if(matchCore(s,p + 2))
return true;
s++;
}
//*不再重复,即直接跳过*
return matchCore(s,p + 2);
} }
};
注:
1.该方法中while循环的部分相当于对*进行匹配1次或多次,而matchCore(s,p+2)相当于忽略* ,即匹配0次。下面来详细阐述一下此处过程:
(1)首先,先判断p和s的当前字符是否相同,如果不相同的话,* 这个字符就可以忽略了,相当于匹配0次,即执行matchCore(s,p+2);
(2)如果p和s的当前字符相同的话,此时要考虑对*进行处理了,即匹配0次、1次还是多次?首先先尝试匹配0次,如果匹配0次的话,p和s的剩下部分都相匹配的话,那么直接就可以返回true了,if(machCore(s,p+2) retrun true;这两行代码就是这个意思。如果匹配0次的话不可行怎么办?此时就是s++相当于匹配1次(如果if(matchCore(s,p+2)这个条件成立的话,相当于匹配1次就成功了),匹配一次完之后再进行判断p和s的当前字符是否相同,然后重复之前的过程;这样就完成了匹配0次、1次或多次这整个过程。
2.一定要注意递归函数的终止条件!!
3.此处有个c++方法需要注意,即string.c_str()的用法,它的功能是将一个string类型的字符串转换成一char *字符串。
方法2(全部递归):
class Solution {
public:
bool isMatch(string s, string p) {
return matchCore(s.c_str(),p.c_str());
}
bool matchCore(const char *s,const char *p)
{
if(*p == '\0' && *s == '\0')
return true;
if(*p == '\0' && *s != '\0')
return false;
//如果下一个字符不是*的话
if(*(p+1) != '*')
{
if(*p == *s||(*p == '.' && *s != '\0'))
return matchCore(s + 1,p + 1);
else//一旦有一个字符不一样,则不匹配
return false;
}
else//下一个字符是*
{ if(*p == *s || (*p == '.' && *s != '\0'))//只有当当前字符匹配成功时,才考虑*重复多少次,否则不用考虑*了,直接跳过
return matchCore(s,p + 2) || matchCore(s + 1,p + 2) || matchCore(s + 1,p); //*重复0次、一次或多次
else
return matchCore(s, p + 2);
} }
};
注:
全部递归的话,比之前的"递归+迭代"法满了不少,方法1用了23ms,但方法2(本方法)用了731ms。所以能不用递归就不用递归,这句话是很有道理的;但递归法写的代码确实看上去更加容易理解,虽然相比迭代有时会爆栈或超时(超时很有可能是添加了多余的操作,有时这种多余的操作并不那么的明显)
方法3(全部递归-优化):
class Solution {
public:
bool isMatch(string s, string p) {
return matchCore(s.c_str(),p.c_str());
}
bool matchCore(const char *s,const char *p)
{
if(*p == '\0' && *s == '\0')
return true;
if(*p == '\0' && *s != '\0')
return false;
//如果下一个字符不是*的话
if(*(p+1) != '*')
{
if(*p == *s||(*p == '.' && *s != '\0'))
return matchCore(s + 1,p + 1);
else//一旦有一个字符不一样,则不匹配
return false;
}
else//下一个字符是*
{ if(*p == *s || (*p == '.' && *s != '\0'))//只有当当前字符匹配成功时,才考虑*重复多少次,否则不用考虑*了,直接跳过
return matchCore(s,p + 2) || matchCore(s + 1,p); //*重复匹配s中字符0次、一次或多次
else
return matchCore(s, p + 2);//如果当前字符不相同的话,把该字符和*一块去掉(相当于重复0次),去匹配pattern+2后的部分
} }
};
方法3相比方法2,主要是优化了一处,即
return matchCore(s,p + 2) || matchCore(s + 1,p + 2) || matchCore(s + 1,p);
变成了
return matchCore(s,p + 2) || matchCore(s + 1,p);
因为仔细想想,重复多次是由多个重复1次组成的啊,所以最终不用再强调重复一次了,直接用matchCore(s+1,p)既可以表示重复1次也可以表示成重复多次。
那么去除这一步后,耗时多少呢?是26ms!!可见递归并不比迭代慢,主要是是否不增加多余的操作。话说,做到这一步心里很是挺爽的~~
此外,还有需要注意的一点:对于任何字符串处理的题目,首先都需要判断该字符串是否为nullptr(空指针)或者为""的情况,这种特例一定是要考虑的!!(对于char *类型的字符串,这两种情况都需要考虑;对于string类型的字符串,一般只考虑""的情况)
LeetCode——10. Regular Expression Matching的更多相关文章
- leetcode 10 Regular Expression Matching(简单正则表达式匹配)
最近代码写的少了,而leetcode一直想做一个python,c/c++解题报告的专题,c/c++一直是我非常喜欢的,c语言编程练习的重要性体现在linux内核编程以及一些大公司算法上机的要求,pyt ...
- Leetcode 10. Regular Expression Matching(递归,dp)
10. Regular Expression Matching Hard Given an input string (s) and a pattern (p), implement regular ...
- leetcode 10. Regular Expression Matching 、44. Wildcard Matching
10. Regular Expression Matching https://www.cnblogs.com/grandyang/p/4461713.html class Solution { pu ...
- [LeetCode] 10. Regular Expression Matching 正则表达式匹配
Given an input string (s) and a pattern (p), implement regular expression matching with support for ...
- LeetCode (10): Regular Expression Matching [HARD]
https://leetcode.com/problems/regular-expression-matching/ [描述] Implement regular expression matchin ...
- [LeetCode] 10. Regular Expression Matching
Implement regular expression matching with support for '.' and '*'. DP: public class Solution { publ ...
- Java [leetcode 10] Regular Expression Matching
问题描述: Implement regular expression matching with support for '.' and '*'. '.' Matches any single cha ...
- [leetcode]10. Regular Expression Matching正则表达式的匹配
Given an input string (s) and a pattern (p), implement regular expression matching with support for ...
- 蜗牛慢慢爬 LeetCode 10. Regular Expression Matching [Difficulty: Hard]
题目 Implement regular expression matching with support for '.' and '*'. '.' Matches any single charac ...
- [LeetCode] 10. Regular Expression Matching ☆☆☆☆☆
Implement regular expression matching with support for '.' and '*'. '.' Matches any single character ...
随机推荐
- 百练1041-反反复复-2016正式C题
C:反反复复 总时间限制: 1000ms 内存限制: 65536kB 描述 Mo和Larry发明了一种信息加密方法.他们首先决定好列数,然后将信息(只包含字母)从上往下依次填入各列,并在末尾补充一 ...
- Java中的关键字
1)48个关键字:abstract.assert.boolean.break.byte.case.catch.char.class.continue.default.do.double.else.en ...
- (15)模型层-什么是ORM
ORM是什么 1.MVC或者MTV框架中包括一个重要的部分,就是ORM,它实现了数据模型与数据库的解耦,即数据模型的设计不需要依赖于特定的数据库,通过简单的配置就可以轻松更换数据库,这极大的减轻了开发 ...
- UVALive-6540 Fibonacci Tree
#include<bits/stdc++.h> using namespace std; int n,m; struct edge { int x; int y; int len; }ed ...
- 《DSP using MATLAB》Problem 5.11
代码: %% ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ %% Output In ...
- Python IDLE theme
#转自 http://www.2cto.com/os/201507/418532.html #win10+python3.5.2 #保护视力 .idlerc 目录下新建名为 config-highli ...
- 【BZOJ4554】【TJOI2016】【HEOI2016】游戏
我好弱啊quq 原题: 在2016年,佳缘姐姐喜欢上了一款游戏,叫做泡泡堂.简单的说,这个游戏就是在一张地图上放上若干个炸弹,看 是否能炸到对手,或者躲开对手的炸弹.在玩游戏的过程中,小H想到了这样一 ...
- 【shell编程】之基础知识-输入/输出和重定向
大多数 UNIX 系统命令从你的终端接受输入并将所产生的输出发送回到您的终端.一个命令通常从一个叫标准输入的地方读取输入,默认情况下,这恰好是你的终端.同样,一个命令通常将其输出写入到标准输出,默 ...
- C#获取IIS所有站点及虚拟目录和应用程序(包含名称及详细信息)
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...
- linux 控制结构
一.if 注: 格式1.格式2:一个条件一个命令: 格式3:一个条件两个命令: 格式4:两个条件三个命令,注意条件的写法. 例1: #!/bin/sh#ifTest#to show the metho ...