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# 题解

  1. 判断长度是否满足要求。

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

    • aStr / bStr 分别为 a / b 匹配的字符串。
    • aLen / bLen 分别为 aStr / bStr 的长度。
    • aNum / bNum 分别为 pattern 中 a / b 的个数。
  2. 对于所有的有序对 (aStr, bStr),顺序取 pattern 中的每个字符,判断 value 对应位置是否匹配。

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

csharp 复制代码
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 后,可以提升效率:

csharp 复制代码
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# 的用户
相关推荐
AC使者11 分钟前
#B1630. 数字走向4
算法
冠位观测者15 分钟前
【Leetcode 每日一题】2545. 根据第 K 场考试的分数排序
数据结构·算法·leetcode
向宇it30 分钟前
【从零开始入门unity游戏开发之——unity篇02】unity6基础入门——软件下载安装、Unity Hub配置、安装unity编辑器、许可证管理
开发语言·unity·c#·编辑器·游戏引擎
古希腊掌管学习的神1 小时前
[搜广推]王树森推荐系统笔记——曝光过滤 & Bloom Filter
算法·推荐算法
qystca1 小时前
洛谷 P1706 全排列问题 C语言
算法
yngsqq1 小时前
一键打断线(根据相交点打断)——CAD c# 二次开发
windows·microsoft·c#
古希腊掌管学习的神1 小时前
[LeetCode-Python版]相向双指针——611. 有效三角形的个数
开发语言·python·leetcode
浊酒南街1 小时前
决策树(理论知识1)
算法·决策树·机器学习
就爱学编程1 小时前
重生之我在异世界学编程之C语言小项目:通讯录
c语言·开发语言·数据结构·算法
学术头条1 小时前
清华、智谱团队:探索 RLHF 的 scaling laws
人工智能·深度学习·算法·机器学习·语言模型·计算语言学