【LeetCode刷题】找到字符串中所有字母异位词

给定两个字符串 sp,找到 s中所有 p异位词的子串,返回这些子串的起始索引。不考虑答案输出的顺序。

示例 1:

复制代码
输入: s = "cbaebabacd", p = "abc"
输出: [0,6]
解释:
起始索引等于 0 的子串是 "cba", 它是 "abc" 的异位词。
起始索引等于 6 的子串是 "bac", 它是 "abc" 的异位词。

示例 2:

复制代码
输入: s = "abab", p = "ab"
输出: [0,1,2]
解释:
起始索引等于 0 的子串是 "ab", 它是 "ab" 的异位词。
起始索引等于 1 的子串是 "ba", 它是 "ab" 的异位词。
起始索引等于 2 的子串是 "ab", 它是 "ab" 的异位词。

提示:

  • 1 <= s.length, p.length <= 3 *
  • sp 仅包含小写字母

解题思路:

  1. s 的长度小于 p 的长度,直接返回空列表(不可能存在异位词)。
  2. 分别统计 p 的字符频率和 s 中长度为 len(p) 的初始窗口的字符频率。
  3. 滑动窗口遍历 s:每次窗口右移一位,更新窗口的字符频率,并比较当前窗口与 p 的字符频率是否一致。若一致,则记录当前窗口的起始索引。

Python代码:

python 复制代码
from typing import List
import re


class Solution:
    def findAnagrams(self, s: str, p: str) -> List[int]:
        n, m = len(s), len(p)
        if n < m:
            return []

        cnt_p = [0] * 26
        cnt_s = [0] * 26

        for i in range(m):
            cnt_p[ord(p[i]) - ord('a')] += 1
            cnt_s[ord(s[i]) - ord('a')] += 1

        res = []
        if cnt_p == cnt_s:
            res.append(0)

        for i in range(1, n - m + 1):
            cnt_s[ord(s[i - 1]) - ord('a')] -= 1
            cnt_s[ord(s[i + m - 1]) - ord('a')] += 1
            if cnt_p == cnt_s:
                res.append(i)

        return res


if __name__ == "__main__":
    # 处理输入(格式如:s = "cbaebabacd",p = "abc")
    s_input = input().strip()
    p_input = input().strip()

    # 提取字符串内容
    s = re.search(r'"(.*?)"', s_input).group(1)
    p = re.search(r'"(.*?)"', p_input).group(1)

    solution = Solution()
    result = solution.findAnagrams(s, p)
    print(result)

LeetCode提交代码:

python 复制代码
class Solution:
    from typing import List
    def findAnagrams(self, s: str, p: str) -> List[int]:
        n, m = len(s), len(p)
        if n < m:
            return []
        
        # 初始化字符频率数组(仅小写字母,长度为26)
        cnt_p = [0] * 26
        cnt_s = [0] * 26
        
        # 统计p的字符频率和s的初始窗口(前m个字符)的频率
        for i in range(m):
            cnt_p[ord(p[i]) - ord('a')] += 1
            cnt_s[ord(s[i]) - ord('a')] += 1
        
        res = []
        # 检查初始窗口是否为异位词
        if cnt_p == cnt_s:
            res.append(0)
        
        # 滑动窗口遍历剩余部分
        for i in range(1, n - m + 1):
            # 移除窗口左侧的字符
            cnt_s[ord(s[i-1]) - ord('a')] -= 1
            # 加入窗口右侧的新字符
            cnt_s[ord(s[i + m - 1]) - ord('a')] += 1
            # 检查当前窗口是否为异位词
            if cnt_p == cnt_s:
                res.append(i)
        
        return res
        

程序运行结果如下:

总结

本文介绍了在字符串s中查找所有p的异位词子串的算法。异位词指字母相同但顺序不同的字符串。算法采用滑动窗口法,通过维护两个字符频率数组(分别对应p和s的当前窗口)进行比较。具体步骤包括:1)检查s长度不小于p;2)初始化字符频率数组;3)滑动窗口遍历s,更新窗口频率并比较。时间复杂度为O(n),空间复杂度为O(1)。示例输入如s="cbaebabacd",p="abc"时输出[0,6],验证了算法的正确性。该解法适用于处理小写字母组成的字符串匹配问题。

相关推荐
数据与后端架构提升之路17 分钟前
TeleTron 源码揭秘:如何用适配器模式“无缝魔改” Megatron-Core?
人工智能·python·适配器模式
英英_44 分钟前
MATLAB数值计算基础教程
数据结构·算法·matlab
一起养小猫1 小时前
LeetCode100天Day14-轮转数组与买卖股票最佳时机
算法·leetcode·职场和发展
hele_two1 小时前
快速幂算法
c++·python·算法
l1t2 小时前
利用DeepSeek将python DLX求解数独程序格式化并改成3.x版本
开发语言·python·算法·数独
jllllyuz2 小时前
基于子集模拟的系统与静态可靠性分析及Matlab优化算法实现
算法·matlab·概率论
程序员-King.2 小时前
day143—递归—对称二叉树(LeetCode-101)
数据结构·算法·leetcode·二叉树·递归
BlockChain8882 小时前
字符串最后一个单词的长度
算法·go
爱吃泡芙的小白白2 小时前
深入解析:2024年AI大模型核心算法与应用全景
人工智能·算法·大模型算法
阿崽meitoufa3 小时前
JVM虚拟机:垃圾收集器和判断对象是否存活的算法
java·jvm·算法