leecode/daily/2019-06-09.md
2020-05-22 18:17:19 +08:00

3.7 KiB
Raw Permalink Blame History

每日一题 - Regular Expression Matching

信息卡片

题目描述

Given an input string (s) and a pattern (p), implement regular expression matching with support for '.' and '*'.


'.' Matches any single character.
'*' Matches zero or more of the preceding element.


The matching should cover the entire input string (not partial).

Note:


s could be empty and contains only lowercase letters a-z.
p could be empty and contains only lowercase letters a-z, and characters
like . or *.


Example 1:


Input:
s = "aa"
p = "a"
Output: false
Explanation: "a" does not match the entire string "aa".


Example 2:


Input:
s = "aa"
p = "a*"
Output: true
Explanation: '*' means zero or more of the precedeng element, 'a'.
Therefore, by repeating 'a' once, it becomes "aa".


Example 3:


Input:
s = "ab"
p = ".*"
Output: true
Explanation: ".*" means "zero or more (*) of any character (.)".


Example 4:


Input:
s = "aab"
p = "c*a*b"
Output: true
Explanation: c can be repeated 0 times, a can be repeated 1 time. Therefore
it matches "aab".


Example 5:


Input:
s = "mississippi"
p = "mis*is*p*."
Output: false

本题要求判断给出的字符串和对应的正则是否匹配匹配返回true否则返回false。

本题光从解题来看可以使用api作弊比如用Java字符串的matches方法就可以一步过这道题本题老老实实做的话就需要用到动态规划。基本思路是考察s和p任意从头到两个字符之间的匹配程度有点像编辑距离那道题。

参考答案

class Solution {
    public boolean isMatch(String s, String p) {
      if (s == null || p == null) return false;
 
      // dp[i][j]代表s的前i位字符和p的前j位字符的匹配程度
      // dp[1][2]代表s的第一个字符和p的前两个字符的匹配
      // 如果s的第i个字符和p的第j个相等或者j为.那么di[i][j]的匹配程度取决于dp[i - 1][j - 1]
      // 如果p的第j个字符为*,那么就要考虑*匹配0个1个n个s的第i个字符的情况以及根本匹配不了的情况
      // 根本匹配不了的意思是指p的j - 1个字符和s的第i个字符不相同此时的匹配情况是dp[i][j] = dp[i][j - 2]
      // 然后是p的第j - 1个字符是.或者和s的i个字符相同的情况此时可以匹配0 1 个n个s的第i个字符
      // 0个: dp[i][j] = dp[i][j - 2]
      // 1个: dp[i][j] = dp[i - 1][j - 1]
      // n个: dp[i][j] = dp[i - 1][j]
      // n个的最不好理解可以按照下面的想
      // 我能匹配你n个我肯定匹配了你前面的n-1个也就是dp[i - 1][j]
      // 最后的结果是dp[s.length()][p.length()]
      boolean[][] dp = new boolean[s.length() + 1][p.length() + 1];

      dp[0][0] = true;

      for (int j = 2; j <= p.length(); j++) {
        if (p.charAt(j - 1) ==  '*' && dp[0][j - 2]) dp[0][j] = true;
      }

      for (int i = 1; i <= s.length(); i++) {
        for (int j = 1; j <= p.length(); j++) {
          if (p.charAt(j - 1) == '.' || p.charAt(j - 1) == s.charAt(i - 1)) {
            dp[i][j] = dp[i - 1][j - 1];
          } else if (p.charAt(j - 1) == '*') {
            if (p.charAt(j - 2) != s.charAt(i - 1) && p.charAt(j - 2) != '.') {
              dp[i][j] = dp[i][j - 2];
            } else {
              dp[i][j] = (dp[i][j - 2] || dp[i][j - 1] || dp[i - 1][j]);
            }
          }
        }
      }

      return dp[s.length()][p.length()];
    }
}

其他优秀解答

暂无