定长滑动窗口和不定长滑动窗口

定长滑动窗口,题目一般会给我们长度为k

1.那么我们1先计算[0~k)第一个窗口里面的满足条件的

2.从i=k开始向右滑动窗口for(int i = k;i<n;i++);

这里我们算的是下标i从0往右移动逐渐循环每次遍历右边的k-1个数,

再判断窗口内的左边界[i-k],右边界[i]这两个数,因为我们移除的是左边的加入的是右边的,如果左边的满足条件,那满足条件而计算的一些贡献结果就要减掉,右边的满足条件贡献就要加上,

不用像暴力算法那样每次都要全部判断遍历整个窗口,提高效率

有时候也会需要使用到哈希表。

整数数组中长度为k的数组中所有的元素各不相同。(当题目要求:长度为 k 的子数组中所有元素互不相同)

需要结合滑动窗口和哈希表,unordered_map<数组元素类型,int>=cnt;

cnt<key>=val;key出现的次数,计算出现的次数和出现的元素,用哈希表计数可以快速判断窗口内是否有重复元素

哈希map = 计数+判重复

cpp 复制代码
class Solution {
public:
    long long maximumSubarraySum(vector<int>& nums, int k) {
        int n = nums.size();
        long long current_sum = 0;
        long long max_sum = 0;
        unordered_map<int, int> cnt; // cnt[kay]=val;数字key出现的次数
        long long unique_num = 0;
        for (int i = 0; i < k; i++) {
            if (cnt[nums[i]] == 0) {
            //如果之前没出现过,说明是新的唯一元素,unique_num 才会 +1。
                unique_num++;
            }
            cnt[nums[i]]++;//把当前元素 nums[i] 加入哈希表,它的出现次数 +1
            current_sum += nums[i];
            if (unique_num == k) {
                max_sum = current_sum;
            }
           
        }
        for (int i = k; i < n; i++) {
            cnt[nums[i - k]]--;
            if (cnt[nums[i - k]] == 0)
                unique_num--;
            current_sum -= nums[i - k];

            if (cnt[nums[i]] == 0)
                unique_num++;
            cnt[nums[i]]++;
            current_sum += nums[i];
            if (unique_num == k) {
                max_sum = max(max_sum, current_sum);
            }
        }
        return max_sum;
    }
};

不定长窗口(最长无重复子串)

必须使用set+双指针(left,right)

开始时left和right都是在数组下标为0的位置,但是left指针是不会动的,动的是右指针right。

right++,一直没有重复的数的时候就一直right++,出现重复时,left++,

cpp 复制代码
class Solution {
public:
    int lengthOfLongestSubstring(string s) {
        int n = s.size();
        //两个指针,一开始都在最左边,然后右边的指针往右移动,左边的先不动,如果右边遇到重复的左边的再动
        unordered_set<char>ooc;
        int max_len=0;
        int left = 0;
        for(int right=0;right<n;right++)
        {   
            while(ooc.count(s[right])){
                ooc.erase(s[left]);
                left++;
            }ooc.insert(s[right]);
            max_len=max(max_len,right-left+1);
        }
    return max_len;
    }
};

不定长滑动窗口也会用到哈希map,这时候就要结合使用哈希map和left,right指针

  1. 判断哈希表中键的数量 while(cnt.size() > ....) //限制种类

  2. 判断哈希表中值的数量 while( cnt[x] >1 );//无重复的判断

这里是针对题目给的数组有具体的数字要求

  1. 去除某个数值 cnt.erase(数组名[下标]);

数量变为0时删掉他,保证不影响cnt,size(), if (cnt[...] == 0) cnt.erase(...);

  1. 判断是否会有重复 while(cnt[数组名[数组下标]] > 1){}

//一般是用下标判断

1和4的逻辑是一样的

cpp 复制代码
class Solution {
public:
    int totalFruit(vector<int>& fruits) {
        int n = fruits.size();
        int left = 0;
       
        int max_num = 0;
        unordered_map<int, int>cnt;
        for (int right = 0; right < n; right++)
        {
            cnt[fruits[right]]++;//这里直接加就好了
            current_num++;
            while (cnt.size() > 2)//判断哈希表中键的数量
            {
                cnt[fruits[left]]--;
                if (cnt[fruits[left]] == 0)//彻底减为0时再去除,不然会影响
                {
                    cnt.erase(fruits[left]);

                }
               
                left++;//左指针再向右移
            }

            max_num = max(max_num, right - left + 1);
        }
        return max_num;
    }
};

含有 若干不同元素

cpp 复制代码
 while(cnt[nums[right]]>1)
            {
                cnt[nums[left]]--;
                current_sum-=nums[left];
                if(cnt[nums[left]]==0){
                    cnt.erase(nums[left]);
                }
                left++;
            }

这里是一个数组中所有元素的频率都 小于等于 k

cpp 复制代码
class Solution {
public:
    int maxSubarrayLength(vector<int>& nums, int k) {
        int n = nums.size();
        int left=0;
        int max_len =0;
        unordered_map<int,int>cnt;
        for(int right=0;right< n;right++)
        {
            cnt[nums[right]]++;
            while(cnt[nums[right]]>k)//k
            {
                cnt[nums[left]]--;
                if(cnt[nums[left]]==0)
                {
                    cnt.erase(nums[left]);
                }
                left++;
            }
             max_len=max(max_len, right-left+1);
        }
        return max_len;
    }
};
相关推荐
keep intensify2 小时前
单词搜索-
算法·深度优先
mjhcsp2 小时前
C++ 梯度下降法(Gradient Descent):数值优化的核心迭代算法
开发语言·c++·算法
yunyun321232 小时前
跨语言调用C++接口
开发语言·c++·算法
m0_518019482 小时前
C++中的装饰器模式变体
开发语言·c++·算法
xushichao19892 小时前
高性能密码学库
开发语言·c++·算法
m0_518019482 小时前
C++代码混淆与保护
开发语言·c++·算法
m0_569881472 小时前
C++中的智能指针详解
开发语言·c++·算法
blackicexs2 小时前
第九周第三天
算法
自信150413057592 小时前
选择排序算法
c语言·数据结构·算法·排序算法