(nice!!!)(LeetCode 面试经典 150 题 ) 30. 串联所有单词的子串 (哈希表+字符串+滑动窗口)

题目:30. 串联所有单词的子串



思路:哈希表+字符串+滑动窗口,时间复杂度0(mlen)。
因为每个字符串的长度都相等,在维护窗口i~i+m
len的元素出现情况后,可以从i+len处继续往后移动。细节看注释。

C++版本:

cpp 复制代码
class Solution {
public:
    vector<int> findSubstring(string s, vector<string>& words) {
    	// 答案
        vector<int> v;
        int n=s.size();
        int m=words.size();
        int len=words[0].size();
        if(n<m*len) return v;
		// 从0~len-1开始遍历
        for(int i=0;i<len&&i+m*len<=n;i++){
        	// 哈希表
            unordered_map<string,int> mp;
            //把words的元素都+1
            for(auto x:words){
                mp[x]++;
            }
            // 先维护窗口i~i+m*len的元素出现情况
            for(int j=0;j<m;j++){
                string t=s.substr(i+j*len,len);
                mp[t]--;
                if(mp[t]==0){
                    mp.erase(t);
                }
            }
            // 如果哈希表为空,说明刚好匹配
            if(mp.size()==0) v.push_back(i);
            // 因为每个字符串的长度都相等,所以可以从i+len处继续往后移动
            for(int j=i+len;j+m*len<=n;j+=len){
            	// 加入新的字符串
                string t=s.substr(j+(m-1)*len,len);
                mp[t]--;
                if(mp[t]==0) mp.erase(t);
                // 删掉旧的字符串
                t=s.substr(j-len,len);
                mp[t]++;
                if(mp[t]==0) mp.erase(t);
                // 如果哈希表为空,说明刚好匹配
                if(mp.size()==0) v.push_back(j);
            }
        }
        return v;
    }
};

JAVA版本:

java 复制代码
class Solution {
    public List<Integer> findSubstring(String s, String[] words) {
        List<Integer> v=new ArrayList<>();
        int n=s.length();
        int m=words.length;
        int len=words[0].length();
        if(n<m*len) return v;

        for(int i=0;i<len&&i+m*len<=n;i++){
            Map<String,Integer> mp=new HashMap<>();
            for(var x:words){
                mp.merge(x,1,Integer::sum);
            }
            for(int j=0;j<m;j++){
                String t=s.substring(i+j*len,i+(j+1)*len);
                mp.merge(t,-1,Integer::sum);
                if(mp.get(t)==0){
                    mp.remove(t);
                }
            }
            if(mp.isEmpty()) v.add(i);
            for(int j=i+len;j+m*len<=n;j+=len){
                String t=s.substring(j+(m-1)*len,j+m*len);
                mp.merge(t,-1,Integer::sum);
                if(mp.get(t)==0) mp.remove(t);
                t=s.substring(j-len,j);
                mp.merge(t,1,Integer::sum);
                if(mp.get(t)==0) mp.remove(t);
                if(mp.isEmpty()) v.add(j);
            }
        }
        return v;
    }
}

GO版本:

go 复制代码
func findSubstring(s string, words []string) []int {
    v:=[]int{}
    n,m:=len(s),len(words)
    lens:=len(words[0])
    if n<m*lens {
        return v
    }
    for i:=0;i<lens&&i+m*lens<=n;i++ {
        mp := make(map[string]int)
        for j:=0;j<m;j++ {
            mp[words[j]]++
        }
        for j:=0;j<m;j++ {
            t:=s[i+j*lens:i+(j+1)*lens]
            mp[t]--
            if mp[t]==0 {
                delete(mp,t)
            }
        }
        if len(mp) == 0 {
            v = append(v, i)
        }
        for j:=i+lens;j+m*lens<=n;j+=lens {
            t:=s[j-lens:j]
            mp[t]++
            if mp[t]==0 {
                delete(mp,t)
            }
            t=s[j+(m-1)*lens:j+m*lens]
            mp[t]--
            if mp[t] ==0 {
                delete(mp,t)
            }
            if len(mp) == 0 {
                v = append(v, j) 
            }

        }
    }
    return v
}
相关推荐
雨中飘荡的记忆16 小时前
大流量下库存扣减的数据库瓶颈:Redis分片缓存解决方案
java·redis·后端
阿里云云原生17 小时前
5 分钟零代码改造,让 Go 应用自动获得全链路可观测能力
云原生·go
over69718 小时前
从 LLM 到全栈 Agent:MCP 协议 × RAG 技术如何重构 AI 的“做事能力”
面试·llm·mcp
心之语歌19 小时前
基于注解+拦截器的API动态路由实现方案
java·后端
SuperEugene20 小时前
Vue状态管理扫盲篇:如何设计一个合理的全局状态树 | 用户、权限、字典、布局配置
前端·vue.js·面试
blasit20 小时前
笔记:Qt C++建立子线程做一个socket TCP常连接通信
c++·qt·tcp/ip
华仔啊20 小时前
Stream 代码越写越难看?JDFrame 让 Java 逻辑回归优雅
java·后端
ray_liang20 小时前
用六边形架构与整洁架构对比是伪命题?
java·架构
Ray Liang21 小时前
用六边形架构与整洁架构对比是伪命题?
java·python·c#·架构设计
Sailing21 小时前
🚀 别再乱写 16px 了!CSS 单位体系已经进入“计算时代”,真正的响应式布局
前端·css·面试