【数据结构-单调队列】力扣2762. 不间断子数组

给你一个下标从 0 开始的整数数组 nums 。nums 的一个子数组如果满足以下条件,那么它是 不间断 的:

i,i + 1 ,...,j 表示子数组中的下标。对于所有满足 i <= i1, i2 <= j 的下标对,都有 0 <= |numsi1 - numsi2| <= 2 。

请你返回 不间断 子数组的总数目。

子数组是一个数组中一段连续 非空 的元素序列。

示例 1:

输入:nums = 5,4,2,4

输出:8

解释:

大小为 1 的不间断子数组:5, 4, 2, 4

大小为 2 的不间断子数组:5,4, 4,2, 2,4

大小为 3 的不间断子数组:4,2,4

没有大小为 4 的不间断子数组。

不间断子数组的总数目为 4 + 3 + 1 = 8 。

除了这些以外,没有别的不间断子数组。

示例 2:

输入:nums = 1,2,3

输出:6

解释:

大小为 1 的不间断子数组:1, 2, 3

大小为 2 的不间断子数组:1,2, 2,3

大小为 3 的不间断子数组:1,2,3

不间断子数组的总数目为 3 + 2 + 1 = 6 。

csharp 复制代码
class Solution {
public:
    long long continuousSubarrays(vector<int>& nums) {
        int n = nums.size();
        multiset<int> q;
        long long ans = 0;
        int left = 0, right = 0;
        while(right < n){
            q.insert(nums[right]);
            while(*q.rbegin() - *q.begin() > 2){
                q.erase(q.find(nums[left++]));
            }
            ans += right - left + 1;
            right++;
        }
        return ans;
    }
};

时间复杂度:O(nlogn) ,其中 n 为 nums 的长度。
空间复杂度:O(N)

我们可以使用multiset来做这道题,multiset会对容器内元素进行默认升序排列。我们此时定义两个指针left和right。q.rbegin代表q的最右侧也就是最大值,q.begin代表q的最左侧也就是最小值,如果该滑动数组内的最大值和最小值之差大于2,那么就要将left进行右移。

关于ans += right - left + 1。

引用网友的评论

首先:对于某个以numsi为左端点,numsj为右端点的合法不间断子序列(以下称为合法窗口)numsi:j+1,窗口内的所有子数组都是合法的。

其次:对于合法窗口numsi:j+1,其必定能够由另一个合法窗口numsi:j新增numsj得到。

最后:对于窗口numsi:j新增一个元素numsj得到新窗口numsi:j+1的这一过程,一共引入了j-i+1个合法子数组。这里直接举一个很简单的例子辅佐这一结论:

例子:1, 2包含121, 2共1+2=3个有效子数组,新增一个元素得到1, 2, 1后,包含1, 2, 1, 1, 2, 2, 1, 1, 2, 1共1+2+1=6个有效子数组。窗口增大这一过程引入了2(j)-0(i)+1=3个新的有效子数组。

推导:长为1的窗口共包含1个不同的子数组,长为2的窗口共包含1+2=3个不同的子数组,长为n的窗口共包含1+2+...+n个不同的子数组。因此,长度为k-1的窗口在增加至长度为k时,共引入了k个新的不同的子数组。

注意:滑动窗口方法实际上会将所有有效的窗口都枚举一遍,因此当滑到numsi:j+1时,numsi:j中所有的有效子数组都已经被统计过了,我们只需要将新增的累计进去就好。

单调队列

csharp 复制代码
class Solution {
public:
    long long continuousSubarrays(vector<int>& nums) {
        deque<int> queMax, queMin;
        int n = nums.size();
        int left = 0, right = 0;
        long long ret = 0;
        while(right < n){
            while(!queMax.empty() && nums[right] > queMax.back()){
                queMax.pop_back();
            }
            while(!queMin.empty() && nums[right] < queMin.back()){
                queMin.pop_back();
            }

            queMax.push_back(nums[right]);
            queMin.push_back(nums[right]);

            while(!queMax.empty() && !queMin.empty() && queMax.front() - queMin.front() > 2){
                if(queMax.front() == nums[left]){
                    queMax.pop_front();
                }

                if(queMin.front() == nums[left]){
                    queMin.pop_front();
                }
                left++;
            }
            ret += right - left + 1;
            right++;
        }
        return ret;
    }
};

时间复杂度和空间复杂度双O(N)

该题的做法参考主页力扣1438

相关推荐
_清歌4 小时前
DSpark 深度解读:DeepSeek-V4 如何用「半自回归」把推理速度提升 85%
算法
统计实现局4 小时前
SVD 的三步走:双对角化、Givens 收敛、排序
算法
躬行见万象4 小时前
《VLA 系列》UniLab 强化训练 | G1 机器人 |复现
算法
统计实现局4 小时前
对称不定分解(Bunch-Kaufman):为什么 Cholesky 不够用
算法
统计实现局4 小时前
dqrsl 拆解:拿着 QR 结果能算出哪 5 种东西
算法
统计实现局4 小时前
为什么 Cholesky 求逆比 Gauss-Jordan 快一倍——行列式溢出防护详
算法
To_OC15 小时前
LC 994 腐烂的橘子:人人都说是 BFS 入门题,我却写了三遍才过
javascript·算法·leetcode
金銀銅鐵19 小时前
[Python] 扩展欧几里得算法
python·数学·算法
To_OC21 小时前
LC 200 岛屿数量:经典 DFS 入门题,我第一次写居然连方向都搞错了
javascript·算法·leetcode