LeetCode 面试题 16.18. 模式匹配

LeetCode 面试题 16.18. 模式匹配

文章目录

一、题目

你有两个字符串,即 patternvaluepattern 字符串由字母 "a""b" 组成,用于描述字符串中的模式。例如,字符串 "catcatgocatgo" 匹配模式 "aabab"(其中 "cat""a""go""b"),该字符串也匹配像 "a""ab""b" 这样的模式。但需注意 "a""b" 不能同时表示相同的字符串。编写一个方法判断value字符串是否匹配pattern字符串。

示例 1:

输入: pattern = “abba”, value = “dogcatcatdog”
输出: true

示例 2:

输入: pattern = “abba”, value = “dogcatcatfish”
输出: false

示例 3:

输入: pattern = “aaaa”, value = “dogcatcatdog”
输出: false

示例 4:

输入: pattern = “abba”, value = “dogdogdogdog”
输出: true
解释: “a”=“dogdog”,b=“”,反之也符合规则

提示:

  • 1 <= len(pattern) <= 1000
  • 0 <= len(value) <= 1000
  • 你可以假设 pattern 只包含字母 "a""b"value 仅包含小写字母。

二、C# 题解

判断长度是否满足要求。
即判断是否存在一对 (aStr, bStr),使得 aLen * aNum + bLen * bNum = value.Length。其中

  • aStr / bStr 分别为 a / b 匹配的字符串。
  • aLen / bLen 分别为 aStr / bStr 的长度。
  • aNum / bNum 分别为 pattern 中 a / b 的个数。

对于所有的有序对 (aStr, bStr),顺序取 pattern 中的每个字符,判断 value 对应位置是否匹配。

代码中用 lens 存储所有的有序对 (aStr, bStr)

public class Solution {
    public bool PatternMatching(string pattern, string value) {
        int n    = pattern.Length, l = value.Length;   // pattern 长度
        int aNum = pattern.Sum(a => a == 'a' ? 1 : 0); // pattern 中 a 的个数
        int bNum = n - aNum;                           // pattern 中 b 的个数

        List<int[]> lens = new List<int[]>(); // lens[i][0]/lens[i][1] 分别为 a/b 匹配的字符串长度
        if (aNum == 0) {
            if (l % bNum != 0) return false;
            lens.Add(new[] { 0, l / bNum });
        }
        else if (bNum == 0) {
            if (l % aNum != 0) return false;
            lens.Add(new[] { l / aNum, 0 });
        }
        else
            for (int i = 0; i <= l / aNum; i++)
                if ((l - i * aNum) % bNum == 0)
                    lens.Add(new[] { i, (l - i * aNum) / bNum });

        foreach (var len in lens) {           // 每种 a/b 对应的长度都试一遍
            bool      ans   = true;           // 记录该种长度下是否匹配成功
            string?[] abStr = { null, null }; // 记录 a/b 对应匹配的字符串
            int       index = 0;              // 当前 value 中的匹配位置
            for (int j = 0; j < pattern.Length; j++) {
                int    ab    = pattern[j] - 'a';              // 1 表示 a,2 表示 b
                int    abLen = len[ab];                       // 取出 a_b 对应的匹配字符串长度
                string sub   = value.Substring(index, abLen); // 查看 value 中的字符串
                index += abLen;                               // 更新位置
                if (abStr[ab] == null) {                      // 之前未出现,则用 abStr 记录下来
                    if (abStr[1 - ab] == sub) return false;   // 如果和另一个字符匹配的字符串一样,则返回 false
                    abStr[ab] = sub;
                }
                else if (abStr[ab] != sub) { // 匹配失败
                    ans = false;             // 该情况返回 false
                    break;
                }
            }
            if (ans) return true; // 如果该情况匹配成功,直接返回 true
        }

        return false;
    }
}
  • 时间:68 ms,击败 66.67% 使用 C# 的用户
  • 内存:36.72 MB,击败 33.33% 使用 C# 的用户

优化掉 lens 后,可以提升效率:

public class Solution {
    public bool PatternMatching(string pattern, string value) {
        int n    = pattern.Length, l = value.Length;   // pattern 长度
        int aNum = pattern.Sum(a => a == 'a' ? 1 : 0); // pattern 中 a 的个数
        int bNum = n - aNum;                           // pattern 中 b 的个数

        if (aNum == 0) return Repeat(value, bNum);
        if (bNum == 0) return Repeat(value, aNum);

        for (int aLen = 0; aLen <= l / aNum; aLen++) {
            if ((l - aLen * aNum) % bNum != 0) continue;
            int       bLen  = (l - aLen * aNum) / bNum;
            string?[] abStr = { null, null };
            int       index = 0;
            bool      ans   = true;
            for (var i = 0; i < pattern.Length; i++) {
                int    ab    = pattern[i] - 'a';
                int    abLen = ab == 0 ? aLen : bLen;
                string sub   = value.Substring(index, abLen);
                index += abLen;
                if (abStr[ab] == null) {                    
                    if (abStr[1 - ab] == sub) return false; 
                    abStr[ab] = sub;
                }
                else if (abStr[ab] != sub) { 
                    ans = false;             
                    break;
                }
            }
            if (ans) return true;
        }
        return false;
    }

    // 判断 value 是否重复 n 次
    public bool Repeat(string value, int n) {
        if (value.Length % n != 0) return false;
        int l = value.Length / n;
        for (int i = 0; i < l; i++)
        for (int j = 1; j < n; j++)
            if (value[(j - 1) * l + i] != value[j * l + i])
                return false;

        return true;
    }
}
  • 时间:60 ms,击败 100.00% 使用 C# 的用户
  • 内存:36.51 MB,击败 66.67% 使用 C# 的用户