leetcode 3013

3013: 将数组分成最小总代价的子数组Ⅱ

第一段的第一个数是确定的,即 nums[0]。

如果知道了第二段的第一个数的位置(记作 p),第三段的第一个数的位置,......,第 k 段的第一个数的位置(记作 q),那么这个划分方案也就确定了。

考虑到 q−p≤dist,本题相当于++在一个大小固定为 dist+1 的滑动窗口内,求前 k−1 小的元素和++。

用两个有序集合来做:

  1. 初始化两个有序集合 L 和 R。注意:为方便计算,把 k 减 1。
  2. 把 nums[1] 到 nums[dist+1] 加到 L 中。
  3. 保留 L 最小的 k 个数,把其余数丢到 R 中。
  4. 从 i=dist+2 开始滑窗。
  5. 先把 out=nums[i−dist−1] 移出窗口:如果 out 在 L 中,就从 L 中移除,否则从 R 中移除。
  6. 然后把 in=nums[i] 移入窗口:++如果 in 小于 L 中的最大元素++,则加入 L,否则加入 R。
  7. 上面两步做完后,如果 L 中的元素个数小于 k(等于 k−1),则从 R 中取一个最小元素加入 L;反之,如果 L 中的元素个数大于 k(等于 k+1),则从 L 中取一个最大元素加入 R。

上述过程维护 L 中元素之和 sumL,取 sumL 的最小值,即为答案。

复制代码
sort(nums.begin()+1,nums.end()); //排序
long long sum=reduce(nums.begin()+1,nums.begin()+dist+2,0LL); //求和
multiset<int> L(nums.begin()+1,nums.begin()+dist+2)

multiset<int> :默认升序,允许重复值。

  • nums.begin()+1:起始迭代器(跳过第 0 个元素,从索引 1 开始)

  • nums.begin()+dist+2 :结束迭代器(不包含 该位置,实际累加到索引 dist+1

    int x=*L.rbegin();
    int x=*R.begin();

复制代码
class Solution {
public:
    long long minimumCost(vector<int>& nums, int k, int dist) {
        k--;
        //sumL,nums[1]到nums[dist+1]之和
        long long sum=reduce(nums.begin()+1,nums.begin()+dist+2,0LL);
        multiset<int> L(nums.begin()+1,nums.begin()+dist+2),R;
        auto L2R=[&](){
            int x=*L.rbegin();
            sum-=x;
            L.erase(L.find(x));
            R.insert(x);
        };
        auto R2L=[&](){
            int x=*R.begin();
            sum+=x;
            R.erase(R.find(x));
            L.insert(x);
        };
        while(L.size()>k) L2R(); //保留 L 最小的 k 个数(注意开头已经把 k 减 1)
        
        long long ans=sum;
        for(int i=dist+2;i<nums.size();i++){
            //移除 out
            int out=nums[i-dist-1];
            auto it=L.find(out);
            if(it!=L.end()){ //如果 out 在 L 中
                sum-=out;
                L.erase(it);
            }
            else R.erase(R.find(out));
            //添加 in
            int in=nums[i];
            if(in<*L.rbegin()){ //如果 in 小于 L 中的最大元素
                sum+=in;
                L.insert(in);
            }
            else R.insert(in);
            //维护大小
            if(L.size()<k) R2L();
            else if(L.size()>k) L2R();
            ans=min(ans,sum); //维护 L 中元素之和 sumL,取 sumL 的最小值
        }

        return ans+nums[0];
    }
};
相关推荐
csuzhucong2 分钟前
力扣OJ(2301-2600)
算法·leetcode·职场和发展
玛丽莲茼蒿6 分钟前
Leetcode hot100 旋转图像【中等】
算法·leetcode·职场和发展
阿Y加油吧11 分钟前
堆 / 优先队列专题二刷笔记:前 K 个高频元素 & 数据流的中位数
java·笔记·算法
Chase_______11 分钟前
LeetCode 2090 题解:半径为 k 的子数组平均值,定长滑动窗口经典题一文搞懂
算法·leetcode·职场和发展
MicroTech202512 分钟前
微算法科技(NASDAQ :MLGO)量子图像加权平均滤波:以量子优势重构图像处理效率与精度
科技·算法·重构
jieyucx13 分钟前
Go 语言函数入门:定义、参数、返回值
c++·算法·golang·入门·函数
Brilliantwxx13 分钟前
【C++】认识标准库STL(1)
开发语言·c++·笔记·程序人生·算法
啦啦啦_999917 分钟前
2. 梯度下降算法分类 & 梯度下降与正规方程对比
人工智能·算法·分类
CHANG_THE_WORLD18 分钟前
gc.py 功能介绍:PDF 对象流还原工具(用于 pdfium 测试)
算法·pdf
li星野19 分钟前
链表通关八题:从反转链表到两数相加,手撕LeetCode高频题
数据结构·学习·leetcode·链表