2021-07-02:正则表达式匹配。给定一个字符串s和一个匹配串p。".“匹配单个字符。”"匹配左边元素的多个字符。判断p是否匹配s。比如s=“ab”,p=“a.”,返回true。比如s=“ab”,p="a",返回false。比如s=“aaa”,p=“a*”,返回true。比如s=“moonfdd”,p="kmoonfdd",返回true,因为""表示零个或者多个,这里’k’表示0个。

福大大 答案2021-07-02:

为了更好的处理边界问题。s和p都追加"1"。比如s="",p=“cc”,加1后s=“1”,p=“cc1”。方法1递归和方法2动态规划都会用到。
1.自然智慧,递归。会递归就行,思想很重要。会了递归,动态规划也就会了。
si指针指向s中某个位置,pi指针指向p中某个位置。
1.1.pi+1不带星。
si指针右移1位,pi指针右移1位。
1.2.pi+1带星。
si指针右移1位,pi指针右移2位。匹配的时候。
si指针右移1位,pi指针右移0位。匹配的时候。
si指针右移0位,pi指针右移2位。匹配的时候和不匹配的时候。
2.动态规划。时间复杂度是O(MN),空间复杂度是O(MN)。

代码用golang编写。代码如下:

package main

import "fmt"

func main() {
s := "moonfdd"
p := "c*c*moonfddc*c*"
ret := isMatch(s, p)
fmt.Println(ret)
} //递归
func isMatch(s string, p string) bool {
s = s + "1"
p = p + "1"
return process(s, 0, p, 0)
} func process(s string, si int, p string, pi int) bool {
if si == len(s) && pi == len(p) {
return true
}
if si == len(s) || pi == len(p) {
return false
}
//pi+1是否是*
if pi+1 < len(p) && p[pi+1] == '*' {
if p[pi] == '.' || p[pi] == s[si] {
if process(s, si+1, p, pi) {
return true
}
if process(s, si+1, p, pi+2) {
return true
}
}
if process(s, si, p, pi+2) {
return true
}
} else {
if p[pi] == '.' || p[pi] == s[si] {
if process(s, si+1, p, pi+1) {
return true
}
}
}
return false
} // 动态规划版本 + 斜率优化
func isMatch3(str string, pattern string) bool {
s := str + "1"
p := pattern + "1"
N := len(s)
M := len(p)
dp := make([][]bool, N+1)
for i := 0; i < N+1; i++ {
dp[i] = make([]bool, M+1)
}
dp[N][M] = true
for j := M - 1; j >= 0; j-- {
dp[N][j] = (j+1 < M && p[j+1] == '*') && dp[N][j+2]
}
// dp[0..N-2][M-1]都等于false,只有dp[N-1][M-1]需要讨论
if N > 0 && M > 0 {
dp[N-1][M-1] = s[N-1] == p[M-1] || p[M-1] == '.'
}
for i := N - 1; i >= 0; i-- {
for j := M - 2; j >= 0; j-- {
if p[j+1] != '*' {
dp[i][j] = ((s[i] == p[j]) || (p[j] == '.')) && dp[i+1][j+1]
} else {
if (s[i] == p[j] || p[j] == '.') && dp[i+1][j] {
dp[i][j] = true
} else {
dp[i][j] = dp[i][j+2]
}
}
}
}
return dp[0][0]
}

执行结果如下:


左神java代码

2021-07-02:正则表达式匹配。给定一个字符串s和一个匹配串p。“.“匹配单个字符。“*“匹配左边元素的多个字符。判断p是否匹配s。比如s=“ab“,p=“a.“,返回true。比如s=“ab“的更多相关文章

  1. python练习:编写一个函数isIn,接受两个字符串作为参数,如果一个字符串是另一个字符串的一部分,返回True,否则返回False。

    python练习:编写一个函数isIn,接受两个字符串作为参数,如果一个字符串是另一个字符串的一部分,返回True,否则返回False. 重难点:定义函数的方法.使用str类型的find()函数,可以 ...

  2. 2021.07.02 P1383 高级打字机题解(可持久化平衡树)

    2021.07.02 P1383 高级打字机题解(可持久化平衡树) 分析: 从可以不断撤销并且查询不算撤销这一骚操作可以肯定这是要咱建一棵可持久化的树(我也只会建可持久化的树,当然,还有可持久化并查集 ...

  3. 2021.07.02 UVa1197 多路归并模板

    2021.07.02 UVa1197 多路归并模板 UVA11997 K Smallest Sums - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 分析: 题解 UVA11997 ...

  4. Js判断一个字符串是否包含一个子串

    Js中经常遇到判断一个字符串是否包含一个子串,java语言中有containes的方法,直接调用就可以了.除非引用第三方数据库,Js中没有contains方法. 为了实现更java语言中contain ...

  5. HashTable集合和练习题_计算一个字符串中每一个字符出现的次数

    HashTable集合 /** * java.util.Hashtable<K,V>集合 implement Map<K,V>接口 * Hashtable:底层也是一个哈希表, ...

  6. Java-map-第一题 (Map)利用Map,完成下面的功能: 从命令行读入一个字符串,表示一个年份,输出该年的世界杯冠军是哪支球队。如果该 年没有举办世界杯,则输出:没有举办世界杯。 附:世界杯冠军以及对应的夺冠年份,请参考本章附录。 附录

    第一题 (Map)利用Map,完成下面的功能: 从命令行读入一个字符串,表示一个年份,输出该年的世界杯冠军是哪支球队.如果该 年没有举办世界杯,则输出:没有举办世界杯. 附:世界杯冠军以及对应的夺冠年 ...

  7. 获取一个字符串中每一个字母出现的次数使用map集合

    package 获取字符串中单字符出现次数; import java.util.Scanner; import java.util.TreeMap; /* * 需求:获取一个字符串中每一个字母出现的次 ...

  8. oracle中一个字符串包含另一个字符串中的所有字符

    oracle中一个字符串包含另一个字符串中的所有字符 --解决监理报告中所勾选的标段信息,与该用户所管理的标段字符串不匹配的问题. select * from a where instr(a,b)&g ...

  9. Java 一个字符串在另外一个字符串出现次数

    统计一个字符串在另外一个字符串出现次数 代码如下: package me.chunsheng.javatest; import java.util.regex.Matcher; import java ...

  10. String 类中的几个练习--获取指定字符串中,大写字母、小写字母、数字的个数||获取一个字符串中,另一个字符串出现的次数

    package cn.homework.demo1; public class GetCount { /* * 获取一个字符串中,另一个字符串出现的次数 * 思想: * 1. indexOf到字符串中 ...

随机推荐

  1. 06-Spring整合mybatis实现简易登录

    1. 文件结构 pojo-Users: //属性名与数据库列名一致 public class Users implements Serializable { private int uid; priv ...

  2. leetcode medium 记录 51-100

      # Title Solution Acceptance Difficulty Frequency     92 Reverse Linked List II       40.3% Medium ...

  3. opengauss集群安装报错

    DB_VERSION:opengauss 3.0.3 1.[GAUSS-51632] [GAUSS-51632] : Failed to do gs_sshexkey.Error: Please en ...

  4. 使用Git进行版本控制,不同的项目怎么设置不同的提交用户名和邮箱呢?

    1.全局设置用户名和邮箱 因为平时除了开发公司项目还会写自己的项目或者去维护开源项目,一般情况下,公司会要求提交代码时使用自己的真名或者拼音和公司邮箱,以前就只会设置全局用户名或邮箱如下 git co ...

  5. 基于声网 Flutter SDK 实现互动直播

    前言 互动直播是实现很多热门场景的基础,例如直播带货.秀场直播,还有类似抖音的直播 PK等.本文是由声网社区的开发者"小猿"撰写的Flutter基础教程系列中的第二篇,他将带着大家 ...

  6. flex弹性盒子中flex-grow与flex的区别

    ​大家在使用flex布局的时候很多情况下都会用到flex-grow这个属性, flex-grow 属性用于设置父元素剩余空间的瓜分比例, flex 属性是 flex-grow.flex-shrink  ...

  7. Laf v1.0 发布:函数计算只有两种,30s 放弃的和 30s 上线的

    一般情况下,开发一个系统都需要前端和后端,仅靠一个人几乎无法胜任,需要考虑的特性和功能非常多,比如: 需要一个数据库来存放数据: 需要一个文件存储来存放各种文件,比如图片文件: 后端需要提供接口供前端 ...

  8. git的Rebase和Merge之间的区别

    有人会说Merge更好,因为它保留了最完整的工作历史.其他人则认为,Rebase变得更整洁,这使审阅者的生活更轻松,更高效.本文将解释合并和重新设置之间的区别是什么,使用它们之一有什么好处. 从根本上 ...

  9. EX6新增的字符串

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  10. [Java]大数运算之加法

    一 题目 Input: String a = "15324564...1455445"; //数字过长,不能转换为int/long型运算 String b = "4564 ...