leetcode 3578

3578: 统计极差最大为K的分割方式数

思路:单调队列+动态规划

单调队列是一种特殊的队列数据结构,它维护队列中的元素具有单调性(单调递增或单调递减)。它主要用于解决++滑动窗口类问题++ ,可以在O(n)时间内解决一些看似需要O(n²)时间的问题,适合处理需要维护区间最值的问题。

单调队列的核心思想是:

  • 维护一个双端队列(deque)

  • 队列中的元素保持单调性

  • 及时移除不再有用的元素

  • 保证队列头部始终是当前窗口的最值

0 <= k <= 10^9

一般定义 f[i+1] 表示++前缀++ nums[0] 到 nums[i] 在题目约束下,分割出的最少(最多)子数组个数,本题是定义成分割方案数。

动态规划:我们设 dp[i+1] 表示以 i 为结尾的前缀子数组 nums[0⋯i] 满足题目要求的分割方案数,++初始化 dp[0]=1,此时前缀为空也看作为一种合法的分割。++ [可分割成一个或多个 非空 的连续子段]枚举最后一个子数组的左端点 j,那么问题变成前缀 nums[0] 到 nums[j−1] 在题目约束下的分割方案数,即 f[j]。当子数组++右端点 i 固定时++ ,由于子数组越长,最大值越大,最小值越小,最大最小的差值越可能大于 k。所以符合要求的++左端点 j++ 一定在一个连续区间 [L,i] 中。累加 f[j] 得递归,分割方案数求和

答案为 f[n]。

复制代码
deque<int> min_q, max_q;

两个完全独立的双端队列,同步地 维护一个滑动窗口的最小值最大值(即"单调队列"最经典的用法)

复制代码
while(!min_q.empty() && x<=nums[min_q.back()]){ 
    min_q.pop_back();
}
min_q.push_back(i);
while(!max_q.empty() && x>=nums[max_q.back()]){
    max_q.pop_back();
}
max_q.push_back(i);
  • min_q:把"大于等于 x 的"从尾端踢掉 → 队列里严格单调递增(头小尾大)。

  • max_q:把"小于等于 x 的"从尾端踢掉 → 队列里严格单调递减(头大尾小)。

所以
min_q递增队列 ,队头存当前窗口最小值
max_q递减队列 ,队头存当前窗口最大值

复制代码
class Solution {
public:
    int countPartitions(vector<int>& nums, int k) {
        constexpr int MOD=1'000'000'007;
        int n=nums.size();
        deque<int> min_q,max_q; //获得窗口中的最大、最小值
        vector<int> f(n+1); //动态规划,f[0]~f[n]递归
        f[0]=1;
        long long sum_f=0; // 窗口中的 f[i] 之和
        int left=0; 

        for(int i=0;i<n;i++){
            int x=nums[i];
            sum_f+=f[i]; //滑动窗口-入

            while(!min_q.empty() && x<=nums[min_q.back()]){ 
                min_q.pop_back();
            }
            min_q.push_back(i);
            while(!max_q.empty() && x>=nums[max_q.back()]){
                max_q.pop_back();
            }
            max_q.push_back(i);

            while(nums[max_q.front()]-nums[min_q.front()]>k){ //最大值与最小值作差超过k
                sum_f-=f[left]; //滑动窗口-出,类前缀和
                left++;
                if(min_q.front()<left){
                    min_q.pop_front();
                }
                if(max_q.front()<left){
                    max_q.pop_front();
                }
            }
            f[i+1]=sum_f%MOD;
        }

        return f[n];
    }
};
相关推荐
2301_764441337 分钟前
LISA时空跃迁分析,地理时空分析
数据结构·python·算法
东北洗浴王子讲AI8 分钟前
GPT-5.4辅助算法设计与优化:从理论到实践的系统方法
人工智能·gpt·算法·chatgpt
Billlly1 小时前
ABC 453 个人题解
算法·题解·atcoder
玉树临风ives1 小时前
atcoder ABC 452 题解
数据结构·算法
feifeigo1231 小时前
基于马尔可夫随机场模型的SAR图像变化检测源码实现
算法
fengfuyao9852 小时前
基于STM32的4轴步进电机加减速控制工程源码(梯形加减速算法)
网络·stm32·算法
无敌昊哥战神3 小时前
深入理解 C 语言:巧妙利用“0地址”手写 offsetof 宏与内存对齐机制
c语言·数据结构·算法
小白菜又菜3 小时前
Leetcode 2075. Decode the Slanted Ciphertext
算法·leetcode·职场和发展
Proxy_ZZ03 小时前
用Matlab绘制BER曲线对比SPA与Min-Sum性能
人工智能·算法·机器学习
黎阳之光3 小时前
黎阳之光:以视频孪生领跑全球,赋能数字孪生水利智能监测新征程
大数据·人工智能·算法·安全·数字孪生