【C++习题】15.滑动窗口_串联所有单词的子串

文章目录


题目链接:

30. 串联所有单词的子串


题目描述:


解法

滑动窗口+哈希表

这题和第14题不同的是:

  1. 哈希表不同:hash<string,int>
  2. leftright指针的移动不同:移动的步长是每个单词的长度
  3. 滑动窗口执行的次数不同

C++ 算法代码:

cpp 复制代码
class Solution 
{
    public:
    vector<int> findSubstring(string s, vector<string>& words) 
    {
        vector<int> ret;
        unordered_map<string, int> hash1; // hash1保存 words 里面所有单词的频次
        for(auto& s : words) hash1[s]++; //统计 word 出现的次数
        int len = words[0].size(), m = words.size(); //len 表示每个单词的长度,m 表示单词的总数
        for(int i = 0; i < len; i++) // 执行 len 次
        {
            unordered_map<string, int> hash2; // 维护窗口内单词的频次
            for(int left = i, right = i, count = 0; right + len <= s.size(); 
                right += len)
            {
                // 进窗口 + 维护 count
                string in = s.substr(right, len);//从字符串 s 中提取一个长度为 len 的子串
                //s.substr(right, len) 这是一个字符串函数,用于从字符串 s 中提取子串。
				//right 是当前窗口的起始位置。
				//len 是每个单词的长度。
                hash2[in]++;//并将其存储在 hash2 中,用于统计当前滑动窗口内单词的出现次数
                /*
                hash1 是一个 unordered_map,用于存储 words 列表中每个单词的出现频次。
				hash1.count(in) 检查单词 in 是否存在于 hash1 中。
				如果 in 存在于 hash1 中,hash1.count(in) 返回 1(真)。
				如果 in 不存在于 hash1 中,hash1.count(in) 返回 0(假)。
				hash2[in] <= hash1[in]
				这部分代码检查当前单词 in 在 hash2 中的出现次数是否小于或等于它在 words 列表中的出现次数(即 hash1[in])。
				如果 hash2[in] 小于或等于 hash1[in],说明当前窗口内单词 in 的出现次数没有超过 words 列表中的限制。
				*/
                if(hash1.count(in) && hash2[in] <= hash1[in]) 
                {
                    count++;
                }
                // 判断
                if(right - left + 1 > len * m)
                {
                    // 出窗口 + 维护 count
                    string out = s.substr(left, len); //s.substr(left, len) ,这个子串 out 是当前窗口最左边的单词。
                    /*
                    hash1.count(out) 检查 out 是否存在于 hash1 中
                    hash2[out] <= hash1[out] 检查当前窗口内 out 的出现次数是否小于或等于 words 列表中 out 的出现次数。
                    如果这两个条件都满足,说明 out 是一个有效的单词,当前窗口的有效单词计数 count 需要减少 1。
                    */
                    if(hash1.count(out) && hash2[out] <= hash1[out]) 
                    {
                        count--;
                    }
                    hash2[out]--; //更新 hash2 中 out 的计数:hash2[out]-- 将 out 在 hash2 中的计数减 1,因为 out 已经离开当前窗口
                    left += len;//left += len 将窗口的左边界向右移动 len 个位置,准备进行下一步的窗口滑动
                }
                // 更新结果
                if(count == m) 
                {
                    ret.push_back(left);
                }
            }
        }
        return ret;
    }
};

图解

例如:s = "barfoothefoobarman", words = ["foo","bar"]

  1. hash1[foo]=1,hash1[bar]=1

    len=3,m=2

    in=bar,hash2[bar]++=1

    满足if(hash1.count(in) && hash2[in] <= hash1[in])count++

    right += len

  1. in=foo,hash2[foo]++=1

    满足if(hash1.count(in) && hash2[in] <= hash1[in])count++

    满足if(count == m) ,所以ret.push_back(left);也就是0

    right += len

  1. in=the,hash2[the]++=1

    right += len

  1. in=foo,hash2[foo]++=2

    满足if(right - left + 1 > len * m)out=barout不是有效字符,count不会减少,hash2[out]--;left += len;

    right += len

  1. 后面步骤类似,就不多赘述了
相关推荐
我想进大厂11 分钟前
图论---朴素Prim(稠密图)
数据结构·c++·算法·图论
我想进大厂16 分钟前
图论---Bellman-Ford算法
数据结构·c++·算法·图论
光而不耀@lgy22 分钟前
C++初登门槛
linux·开发语言·网络·c++·后端
lkbhua莱克瓦2427 分钟前
用C语言实现——一个中缀表达式的计算器。支持用户输入和动画演示过程。
c语言·开发语言·数据结构·链表·学习方法·交友·计算器
Mr__Miss31 分钟前
面试踩过的坑
java·开发语言
啊丢_33 分钟前
C++——Lambda表达式
开发语言·c++
Chh07151 小时前
《R语言SCI期刊论文绘图专题计划》大纲
开发语言·r语言
Yeats_Liao1 小时前
Go 语言 TCP 端口扫描器实现与 Goroutine 池原理
开发语言·tcp/ip·golang
Wendy_robot2 小时前
【滑动窗口+哈希表/数组记录】Leetcode 438. 找到字符串中所有字母异位词
c++·算法·leetcode
Thomas_YXQ2 小时前
Unity3D IK解算器技术分析
开发语言·搜索引擎·unity·全文检索·unity3d·lucene