找到字符串中所有字母异位词(medium)

一题目链接

找到字符串中所有字母异位词

二题目描述

三解法(滑动窗口 + 哈希表)

3.1算法思路:

◦ 因为字符串 p 的异位词的长度一定与字符串 p 的长度相同,所以我们可以在字符串 s 中构
造一个长度为与字符串 p 的长度相同的滑动窗口,并在滑动中维护窗口中每种字母的数量;
◦ 当窗口中每种字母的数量与字符串 p 中每种字母的数量相同时,则说明当前窗口为字符串 p
的异位词;
◦ 因此可以用两个大小为 26 的数组来模拟哈希表,一个来保存 s 中的子串每个字符出现的个
数,另一个来保存 p 中每一个字符出现的个数。这样就能判断两个串是否是异位词。

3.2C++ 算法代码:

cpp 复制代码
class Solution
{
    public:
        vector<int> findAnagrams(string s, string p)
        {
            vector<int> ret;
            int hash1[26] = { 0 }; // 统计字符串 p 中每个字符出现的个数
            for(auto ch : p) hash1[ch - 'a']++;
            int hash2[26] = { 0 }; // 统计窗口里面的每一个字符出现的个数
            int m = p.size();
            for(int left = 0, right = 0, count = 0; right < s.size(); right++)
            {
                char in = s[right];
                // 进窗口 + 维护 count
                if(++hash2[in - 'a'] <= hash1[in - 'a']) count++;
                if(right - left + 1 > m) // 判断
                {
                    char out = s[left++];
                    // 出窗口 + 维护 count
                    if(hash2[out - 'a']-- <= hash1[out - 'a']) count--;
                }
                // 更新结果
                if(count == m) ret.push_back(left);
             }
            return ret;
        }
};

3.2.1C++结果

3.3Java 算法代码:

java 复制代码
class Solution
{
    public List<Integer> findAnagrams(String ss, String pp)
    {
        List<Integer> ret = new ArrayList<Integer>();
        char[] s = ss.toCharArray();
        char[] p = pp.toCharArray();
        int[] hash1 = new int[26]; // 统计字符串 p 中每一个字符出现的个数
        for(char ch : p) hash1[ch - 'a']++;
        int[] hash2 = new int[26]; // 统计窗口中每一个字符出现的个数
        int m = p.length;
        for(int left = 0, right = 0, count = 0; right < s.length; right++)
        {
            char in = s[right];
            // 进窗口 + 维护 count
            if(++hash2[in - 'a'] <= hash1[in - 'a']) count++;
            if(right - left + 1 > m) // 判断
            {
                char out = s[left++];
                // 出窗口 + 维护 count
                if(hash2[out - 'a']-- <= hash1[out - 'a']) count--;
            }
            // 更新结果
            if(count == m) ret.add(left);
        }
        return ret;
    }
}

3.3.2Java结果

四细节与避坑

  1. 边界处理 :如果 p.size() > s.size(),直接返回空数组,避免越界。
  2. 数组对比 :用 memcmp 直接对比两个 26 元素的数组,比循环判断更高效。
  3. 窗口滑动逻辑
    • 左边界索引:i - n2
    • 新窗口起始索引:i - n2 + 1
  4. 内存优化:全程不用创建子串,只用两个 26 大小的数组,空间复杂度 O (1),不会超内存。
相关推荐
_日拱一卒1 小时前
LeetCode:230二叉搜索树中第K小的元素
算法
Struggle_97551 小时前
算法知识-背包dp
算法
ZPC82101 小时前
Linux 共享内存(shm)
算法
水木流年追梦1 小时前
大模型入门-应用篇2-RAG (检索增强生成):从原理到 Python 实战
开发语言·python·算法·prompt
数智工坊1 小时前
【连续动作空间算法2】TD3:DDPG的终极进化,根治连续控制中的函数近似误差
论文阅读·人工智能·算法·计算机视觉
handler011 小时前
速通蓝桥杯省一: 前缀和&差分(附经典例题)
c语言·c++·笔记·职场和发展·蓝桥杯
洛水水1 小时前
【力扣100题】25. 搜索二维矩阵 II
算法·leetcode·矩阵
样例过了就是过了1 小时前
LeetCode热题100 多数元素
c++·算法·leetcode·贪心算法
白羊by1 小时前
YOLOv8 官方损失函数详解(按任务分类)
人工智能·深度学习·算法·yolo·分类