力扣-将x减到0的最小操作数

1.题目描述

2.题目链接

1658. 将 x 减到 0 的最小操作数 - 力扣(LeetCode)

3.题目分析

1)正面求解困难

题目要求我们每次都从最左边或者最右边取一个数,使x-元素的值,并在数组中移除该元素。最后返回的最小操作数也就是移除数组元素的最小个数

也就是如下:

其中a和b代表对应子串的元素和。

这道题如果我们从正面求解,是非常困难的。因为他有可能是从最左边移除,也有可能从最右边移除,并且要求a+b=x,还要求最小值,情况是非常多的

2)正难则反

既然正面求解困难,那么我们反过来换一种思路,从反面求解,也就是求和为sum-x的子串的最长长度:

其中sum表示数组所有元素和。

这就和我们前面做过的一道滑动窗口的oj题目非常相似了: 力扣-长度最小的子数组-CSDN博客

4.代码解答

java 复制代码
class Solution {
    public int minOperations(int[] nums, int x) {
        int length=-1,sum=0,temp=0;
        for(int a:nums)sum+=a;
        int target=sum-x;
        if(target<0){
            return -1;
        }
        for(int left=0,right=0;right<nums.length;right++){
            temp+=nums[right];
            while(temp>target&&left<=right){
                temp-=nums[left++];
            }
            if(temp==target){
                length=Math.max(length,right-left+1);
            }
        }
        return length==-1?-1:nums.length-length;
    }
}

5.代码细节

1)length的初始值

java 复制代码
 int length=-1,sum=0,temp=0;

题目中要求如果没有找到符合条件的字串,就返回-1,所以我们通过定义length的初始值为-1,在结合三位运算符进行返回:

java 复制代码
  return length==-1?-1:nums.length-length;

2)while循环的条件

java 复制代码
 while(temp>target&&left<=right){
                temp-=nums[left++];
            }

应该是left<=right而不是left<=right,确保窗口可以收缩到空(left > right),从而正确处理所有边界情况。这时中间的子串为空,也就是说,此时整个数组的元素和是题目中给出的x,要返回的就是整个数组的长度。

这是滑动窗口算法中常见的边界陷阱,需要特别注意!

相关推荐
modelmd20 小时前
【递归算法】汉诺塔
python·算法
2401_8384725120 小时前
C++中的装饰器模式实战
开发语言·c++·算法
白中白1213820 小时前
算法题-06
算法
珍珠是蚌的眼泪20 小时前
LeetCode_二叉树1
leetcode·二叉树·层序遍历·前序遍历·中序遍历·后续遍历
爱学习的阿磊20 小时前
C++与Qt图形开发
开发语言·c++·算法
爱敲代码的TOM20 小时前
基础算法技巧总结2(算法技巧零碎点,基础数据结构,数论模板)
数据结构·算法
惊讶的猫20 小时前
InnoDB选用B+树作为索引的原因
数据结构·b树
liu_endong20 小时前
杰发科技AC7840——打印所有GPIO的PORT配置寄存器
mcu·算法·杰发科技·autochips·车规芯片
老鼠只爱大米20 小时前
LeetCode经典算法面试题 #199:二叉树的右视图(BFS双队列法、DFS递归法等多种实现方案详细解析)
算法·leetcode·二叉树·dfs·bfs·深度优先搜索·右视图
历程里程碑20 小时前
子串-----和为 K 的子数组
java·数据结构·c++·python·算法·leetcode·tornado