leetcode hot100 438. 找到字符串中所有字母异位词 滑动窗口 medium

直觉:
滑动窗口 + 排序法

窗口的大小会始终保持为 p 的长度,保证每次比较的都是相同长度的子串

窗口右移

排序比较

python 复制代码
class Solution:
    def findAnagrams(self, s: str, p: str) -> List[int]:

        if len(s) < len(p):
            return []

        if not s or not p:
            return []

        res = []
        p_sorted = sorted(p)  # 排序 p 以便与窗口进行比较

        # 使用切片获取初始窗口
        for i in range(len(p) - 1, len(s)):
            window = s[i - len(p) + 1: i + 1]  # 正确获取窗口
            if sorted(window) == p_sorted:
                res.append(i - len(p) + 1)  # 存储当前窗口的起始索引

        return res

排序的时间复杂度是 O(k log k),其中 k 是 p 的长度。

每次滑动窗口时需要对窗口进行排序

时间复杂度 :O(n * k log k),其中 n 是字符串 s 的长度,k 是字符串 p 的长度
空间复杂度 :O(n + k),其中 n 是字符串 s 的长度,k 是字符串 p 的长度。

如果字符串 p 较大(即 k 较大),排序操作的代价会比较高

滑动窗口 + 频率计数法

python 复制代码
class Solution:
    def findAnagrams(self, s: str, p: str) -> List[int]:

        cnt_p = Counter(p)  # 统计 p 中每个字符的频率  "abc" -> Counter({'a': 1, 'b': 1, 'c': 1})
        cnt_window = Counter()  # 统计 s 中当前窗口的字符频率
        plen = len(p)

        res = []
        left = 0

        for right in range(len(s)):

            cnt_window[s[right]] += 1  # 右端点字符进入窗口,增加该字符的计数

            winlen = right - left +1
            if winlen < plen:
                continue   # 继续for循环移动右指针

            # 达到p长度
            else:
                if cnt_p == cnt_window:   # # 如果窗口的字符频率和 p 的字符频率一样,说明是异位词
                    res.append(left)  # 记录左端点的索引

                    cnt_window[s[left]] -= 1   # 左端点移出windows
                    left += 1   # 左指针前进
                else :
                    cnt_window[s[left]] -= 1   # 左端点移出windows
                    left += 1   # 左指针前进

        return res

总时间复杂度 :O(n×k)

其中 n 是字符串 s 的长度,k 是字符串 p 的长度。
空间复杂度: O(k+n)

相关推荐
TracyCoder12310 小时前
LeetCode Hot100(19/100)——206. 反转链表
算法·leetcode
测试涛叔10 小时前
金三银四软件测试面试题(800道)
软件测试·面试·职场和发展
踩坑记录10 小时前
leetcode hot100 94. 二叉树的中序遍历 easy 递归 dfs
leetcode
Angelina_Jolie11 小时前
一文搞懂 SCI、SSCI、CSSCI、C 刊、核心期刊:定义、作用、层级对比及投稿选择
考研·职场和发展·创业创新
醉颜凉12 小时前
【LeetCode】打家劫舍III
c语言·算法·leetcode·树 深度优先搜索·动态规划 二叉树
达文汐12 小时前
【困难】力扣算法题解析LeetCode332:重新安排行程
java·数据结构·经验分享·算法·leetcode·力扣
一匹电信狗12 小时前
【LeetCode_21】合并两个有序链表
c语言·开发语言·数据结构·c++·算法·leetcode·stl
User_芊芊君子12 小时前
【LeetCode经典题解】搞定二叉树最近公共祖先:递归法+栈存路径法,附代码实现
算法·leetcode·职场和发展
培风图南以星河揽胜12 小时前
Java版LeetCode热题100之零钱兑换:动态规划经典问题深度解析
java·leetcode·动态规划
算法_小学生12 小时前
LeetCode 热题 100(分享最简单易懂的Python代码!)
python·算法·leetcode