[滑动窗口] 12. 将 x 减到 0 的最小操作数

一. 题目描述

只能从此刻最左或最右的元素选一个被x减并从数组中移除,要求删的元素最少。

二. 解题思路

1)正难则反,删两边找最短不好找,我们就删中间找最长

这样题目就可以转化为:找出最长的子数组,使子数组中所有元素的和恰好等于sum - x (sum是原数组中所有元素之和)。最后用原数组长度减去我们找到的中间最长子数组的长度并返回,如果没有恰好等于sum - x的就返回-1。

2)找子数组我们最先想到的一定是暴力枚举,固定一个数枚举剩下所有的组合。

通过暴力解法我们可以想到一种优化方案--滑动窗口(left和right都持续朝一个方向前进),(target = sum - x,双指针left和right标记子数组)因为只有当加入最新的一个nums[right] 导致当前子数组的和cursum 大于了 target,所以才需要让left++,减小子数组当前的和。因此,right是一定不需要再回到left的位置重新向后遍历的,这些位置的和都是小于target的。

两个指针同向而行,本题的解法就是滑动窗口。

3)right++,cursum += nums[right] 进窗口;每进一个数,判断是否需要出窗口,本题判断的条件是cursum > target就需要出窗口,让cursum -= nums[left],left++; 此时一定可以保证cursum <= target,判断一下当cursum == target时,更新结果。

4)细节处理很重要,易错用例:

五. 代码实现

下面我虽然给出两个版本,但大逻辑是完全一样的就是滑动窗口,只是在细节处理上有一点差别。

cpp 复制代码
class Solution 
{
public:
    int minOperations(vector<int>& nums, int x) 
    {
        // 统计所有元素的和
        int sum = 0;
        for(const auto& num : nums)
            sum += num;

        int left = 0, right = 0, target = sum - x;
        int cursum = 0, maxlen = -1;

        // 找最长子数组,以此保证两边最短
        for(; right < nums.size(); right++)
        {
            // 进窗口
            cursum += nums[right];
            // 判断
            while(left < nums.size() && cursum > target)
                cursum -= nums[left++];
            // 此时一定<=target,等于时更新结果
            if(cursum == target)
                maxlen = max(maxlen, right - left + 1);
        }
        return maxlen == -1 ? maxlen : nums.size() - maxlen;
    }
};
cpp 复制代码
class Solution 
{
public:
    int minOperations(vector<int>& nums, int x) 
    {
        // 统计所有元素的和
        int sum = 0;
        for(const auto& num : nums)
            sum += num;

        int left = 0, right = 0, target = sum - x;
        int cursum = 0, maxlen = -1;

        if(target < 0) return -1;

        // 找最长子数组,以此保证两边最短
        for(; right < nums.size(); right++)
        {
            // 进窗口
            cursum += nums[right];
            // 判断
            while(cursum > target)
                cursum -= nums[left++];
            // 此时一定<=target,等于时更新结果
            if(cursum == target)
                maxlen = max(maxlen, right - left + 1);
        }
        return maxlen == -1 ? maxlen : nums.size() - maxlen;
    }
};
相关推荐
一条大祥脚1 小时前
Codeforces Round 1098 (Div. 2)
算法·深度优先
时空自由民.1 小时前
平衡车PID控制系统(豆包版本)
算法
sno_guo2 小时前
直播抠图技术100谈之25---调色中曲线是最优解
人工智能·算法·机器学习·直播·内容运营·obs抠图·直播技术
故事和你912 小时前
洛谷-【图论2-2】最短路1
开发语言·数据结构·c++·算法·动态规划·图论
Simple-Soft2 小时前
指针的高级应用与技巧 - C语言的灵魂
c语言·数据结构·算法
南宫萧幕2 小时前
Simulink 从零搭建 HEV ECMS 环境:模块解析、排坑指南与智能算法接口预留
人工智能·算法·matlab·汽车·控制
子豪-中国机器人2 小时前
词云与条形码答案
算法
闲人编程2 小时前
Agent的评估体系(AgentEval):如何判断一个Agent好坏?
大数据·人工智能·python·算法·agent·智能体·swe
沫璃染墨2 小时前
红黑树完全指南:从核心原理到插入验证全实现
开发语言·c++·算法