LeetCode 1358.包含所有三种字符的子字符串数目:滑动窗口(两种写法直接推荐方法二)

【LetMeFly】1358.包含所有三种字符的子字符串数目:滑动窗口(两种写法直接推荐方法二)

力扣题目链接:https://leetcode.cn/problems/number-of-substrings-containing-all-three-characters/

给你一个字符串 s ,它只包含三种字符 a, b 和 c 。

请你返回 a,b 和 c 都 至少出现过一次的子字符串数目。

示例 1:

复制代码
输入:s = "abcabc"
输出:10
解释:包含 a,b 和 c 各至少一次的子字符串为 "abc", "abca", "abcab", "abcabc", "bca", "bcab", "bcabc", "cab", "cabc" 和 "abc" (相同字符串算多次)。

示例 2:

复制代码
输入:s = "aaacb"
输出:3
解释:包含 a,b 和 c 各至少一次的子字符串为 "aaacb", "aacb" 和 "acb" 。

示例 3:

复制代码
输入:s = "abc"
输出:1

提示:

  • 3 <= s.length <= 5 x 10^4
  • s 只包含字符 a,b 和 c 。

解题方法:滑动窗口

使用两个指针,时刻满足指针之间的"窗口"是最短的同时包含abc的窗口。

滑动方法一:枚举窗口起点l(l 到 r-1以及r的右边 全合法)------推荐方法二

两个指针,每次左指针 l l l右移一位,不断右移右指针 r r r直到窗口 [ l , r ) [l, r) [l,r)中含有三种字母。

那么对于起点 l l l,其到 r − 1 r-1 r−1合法到话,其到 r r r、 r + 1 r+1 r+1、 ⋯ \cdots ⋯、 n − 1 n-1 n−1(共计 n − r + 1 n-r+1 n−r+1个)都合法。

滑动方法二:枚举窗口终点r(l-1以及l-1的左边 到 r 全合法)

两个指针,每次右指针 r r r右移一位,不断右移左指针 l l l直到首次满足 窗口 l , r l, r l,r中不含有三种字母。

那么 l − 1 , r l-1, r l−1,r l − 2 , r l-2, r l−2,r、 ⋯ \cdots ⋯、 0 , r 0, r 0,r(如果存在)(共计l个)均合法。

时空复杂度分析

  • 时间复杂度 O ( l e n ( s ) ) O(len(s)) O(len(s)),每个字母最多被遍历两次
  • 空间复杂度 O ( 1 ) O(1) O(1)

AC代码

C++------方法二

cpp 复制代码
/*
 * @LastEditTime: 2026-07-01 15:38:27
 */
class Solution {
public:
    int numberOfSubstrings(string s) {
        int ans = 0;
        int cnt[3] = {0};
        for (int l = 0, r = 0, n = s.size(); r < n; r++) {
            cnt[s[r] - 'a']++;
            while (cnt[0] && cnt[1] && cnt[2]) {
                cnt[s[l++] - 'a']--;
            }
            ans += l;
        }
        return ans;
    }
};

C++------方法一

cpp 复制代码
/*
 * @LastEditTime: 2026-07-01 15:34:43
 */
class Solution {
public:
    int numberOfSubstrings(string s) {
        int ans = 0;
        int cnt[3] = {0};
        for (int l = 0, r = 0, n = s.size(); l < n; l++) {
            while (r < n && !(cnt[0] && cnt[1] && cnt[2])) {
                cnt[s[r++] - 'a']++;
            }
            // printf("l = %d, r = %d, cnt = {%d, %d, %d}, ans += %d\n", l, r, cnt[0], cnt[1], cnt[2], n - r + 1);
            ans += (cnt[0] && cnt[1] && cnt[2]) ? n - r + 1 : 0;
            cnt[s[l] - 'a']--;
        }
        return ans;
    }
};

Python------方法二

python 复制代码
'''
LastEditTime: 2026-07-01 15:49:11
'''
class Solution:
    def numberOfSubstrings(self, s: str) -> int:
        ans = 0
        cnt = [0, 0, 0]
        l = r = 0
        for r in range(len(s)):
            cnt[ord(s[r]) - ord('a')] += 1
            while cnt[0] and cnt[1] and cnt[2]:
                cnt[ord(s[l]) - ord('a')] -= 1
                l += 1
            ans += l
        return ans

同步发文于CSDN和我的个人博客,原创不易,转载经作者同意后请附上原文链接哦~

千篇源码题解已开源

相关推荐
QiLinkOS10 小时前
第三视觉理解徐玉生与他的商业活动(28)
大数据·c++·人工智能·算法·开源协议
wabs66610 小时前
关于动态规划【力扣1143.最长公共子序列的思考】
算法·leetcode·动态规划
剑挑星河月11 小时前
54.螺旋矩阵
java·算法·leetcode·矩阵
Robot_Nav11 小时前
MPPI 局部规划器实验设计讲解
人工智能·算法·mppi
mingo_敏12 小时前
Mean-Teacher 均值教师自训练框架详解
算法·均值算法
星空露珠12 小时前
迷你世界UGc3.0脚本Wiki[剧情动画模块管理接口 Timeline]
开发语言·数据结构·算法·游戏·lua
笨笨没好名字12 小时前
Leetcode刷题python3版第一周(下)
linux·算法·leetcode
jinyishu_12 小时前
常见排序算法详解
数据结构·算法·排序算法
手写码匠12 小时前
手写 LLM 安全护栏:从内容审核到越狱防御的完整实现
人工智能·深度学习·算法·aigc
luj_176812 小时前
草酸与烟酸对消化及糖代谢的影响解析
服务器·c语言·开发语言·经验分享·算法