【leetcode刷题笔记】Wildcard Matching
Implement wildcard pattern matching with support for '?' and '*'.
'?' Matches any single character.
'*' Matches any sequence of characters (including the empty sequence). The matching should cover the entire input string (not partial). The function prototype should be:
bool isMatch(const char *s, const char *p) Some examples:
isMatch("aa","a") → false
isMatch("aa","aa") → true
isMatch("aaa","aa") → false
isMatch("aa", "*") → true
isMatch("aa", "a*") → true
isMatch("ab", "?*") → true
isMatch("aab", "c*a*b") → false
题解:昨天晚上写了一晚上的递归,一直TLE。早上果断改用dp了,结果dp还TLE了好多次,真是说多了都是泪。
dp的思想很简单:用二维数组dp[s_length+1][p_length+1]记录结果。
- 首先如果s和p都是空串的话,那么它们是匹配的,所以dp[0][0] = true;
- 当s为空串的时候(dp中的第一行),dp[0][i] = p[i] == '*'? dp[0][i-1]:false;(i=1,...,p_length);
- 对于任意j,如果p(j-1) == '*',dp[i][j]= dp[i-1][j] || dp[i][j-1],对应了两种情况,前一种是不匹配‘*’,后一种情况是匹配'*‘,如下图所示:

- 如果p(j-1) != '*',则只有p(j-1) == '?'或者s(i-1) == p(j-1)的时候才有dp[i][j] = true;否则dp[i][j] = false;
开始以为这样就可以过了,事实证明I am too young too simple。这样还是会超时。要优化两个地方:
- 数出p中不是'*'的字符个数,如果比s的总长度还长,那么s是没有办法匹配的,直接返回false;
- 第二个优化让我非常无语,就是在实现循环的时候,不要每次用p.charAt(j)来取字符,要在开始用ch_p = p.charAt(j)把字符记下来,在以后的循环中就用这个,这么看来p.charAt(j)这个操作还是很耗时的,以后如果在程序中反复使用,都要把它存下来再使用。
最后AC的代码如下:
public class Solution {
public boolean isMatch(String s, String p) {
int m = s.length();
int n = p.length();
int count = 0;
for(int indexP = 0;indexP<n;indexP++)
if(p.charAt(indexP) != '*')
count++;
if(count > m)
return false;
boolean[][] dp =new boolean[m+1][n+1];
dp[0][0] = true;
for(int j = 1;j<=n;j++)
{
char ch_p = p.charAt(j-1);
if(ch_p =='*' && dp[0][j-1])
dp[0][j]= true;
for(int i= 1;i<=m;i++){
char ch_s = s.charAt(i-1);
if(ch_p =='*')
dp[i][j]= dp[i-1][j]|| dp[i][j-1];
else if(ch_p == '?' || ch_p == ch_s){
dp[i][j] = dp[i-1][j-1];
}
else
dp[i][j]= false;
}
}
return dp[m][n];
}
}
【leetcode刷题笔记】Wildcard Matching的更多相关文章
- LeetCode刷题笔记和想法(C++)
主要用于记录在LeetCode刷题的过程中学习到的一些思想和自己的想法,希望通过leetcode提升自己的编程素养 :p 高效leetcode刷题小诀窍(这只是目前对我自己而言的小方法,之后会根据自己 ...
- 18.9.10 LeetCode刷题笔记
本人算法还是比较菜的,因此大部分在刷基础题,高手勿喷 选择Python进行刷题,因为坑少,所以不太想用CPP: 1.买股票的最佳时期2 给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格. ...
- LeetCode刷题笔记 - 12. 整数转罗马数字
学好算法很重要,然后要学好算法,大量的练习是必不可少的,LeetCode是我经常去的一个刷题网站,上面的题目非常详细,各个标签的题目都有,可以整体练习,本公众号后续会带大家做一做上面的算法题. 官方链 ...
- Leetcode刷题笔记(双指针)
1.何为双指针 双指针主要用来遍历数组,两个指针指向不同的元素,从而协同完成任务.我们也可以类比这个概念,推广到多个数组的多个指针. 若两个指针指向同一数组,遍历方向相同且不会相交,可以称之为滑动窗口 ...
- 【leetcode刷题笔记】Regular Expression Matching
Implement regular expression matching with support for '.' and '*'. '.' Matches any single character ...
- LeetCode刷题笔记(1-9)
LeetCode1-9 本文更多是作为一个习题笔记,没有太多讲解 1.两数之和 题目请点击链接 ↑ 最先想到暴力解法,直接双循环,但是这样复杂度为n平方 public int[] twoSum(int ...
- leetcode刷题笔记
(1)Best Time to Buy and Sell Stock Total Accepted: 10430 Total Submissions: 33800My Submissions Say ...
- leetcode刷题笔记08 字符串转整数 (atoi)
题目描述 实现 atoi,将字符串转为整数. 在找到第一个非空字符之前,需要移除掉字符串中的空格字符.如果第一个非空字符是正号或负号,选取该符号,并将其与后面尽可能多的连续的数字组合起来,这部分字符即 ...
- LeetCode刷题笔记-回溯法-分割回文串
题目描述: 给定一个字符串 s,将 s 分割成一些子串,使每个子串都是回文串. 返回 s 所有可能的分割方案. 示例: 输入: "aab"输出:[ ["aa", ...
随机推荐
- python第四周迭代器生成器序列化面向过程递归
第一节装饰器复习和知识储备------------ 第一节装饰器复习和知识储备------------ def wrapper(*args,**kwargs): index(*args,**kwa ...
- Ext扩展的QQ表情选择面板
Ext扩展的QQ表情选择面板 define(function () { EmoteChooser = function(cfg){ this.width=340; this.height=112; t ...
- 我的第六个java程序 spring-bean
配置文件 Beans.xml <?xml version="1.0" encoding="UTF-8"?> <beans xmlns=&quo ...
- [搬家]新域名 akagi201.org
现在感觉自己做了好多年的垃圾信息制造者 以后只在网络上发布有用的东西, 垃圾或者对别人没用的东西就放到自己的硬盘上把 http://akagi201.org
- Package java.sql
Provides the API for accessing and processing data stored in a data source (usually a relational dat ...
- JavaScript 函数语法
函数就是包裹在花括号中的代码块,前面使用了关键词 function: function functionname() { 这里是要执行的代码 } 当调用该函数时,会执行函数内的代码. 可以在某事件发生 ...
- Eclipse 悬浮提示
Eclipse 悬浮提示 使用悬浮提示 java 编辑器中包含了不同类型的悬浮提示,悬浮提示提供了鼠标指针指向元素的额外信息.所有java编辑器中相关的悬浮提示可以通过 preference(首选项) ...
- 电脑出现“损坏的图像”窗口提示dll没有被指定在Windows上运行如何解决
电脑中出现了无法运行应用程序的情况,弹出一个“***.exe - 损坏的图像”的窗口,上面提示“***.dll没有被指定在Windows上运行……”,如果我们遇到这样的问题,应该要如何解决呢? 1.我 ...
- 浅谈DNS
什么叫域名解析 域名解析是把域名指向网站空间IP,让人们通过注册的域名可以方便地访问到网站一种服务.IP地址是网络上标识站点的数字地址,为了方便记忆,采用域名来代替IP地址标识站点地址.域名解析就是域 ...
- JS原生追加子节点
var fragment = document.createDocumentFragment(); li = document.createElement('li'); li.className = ...