Leetcode 将 x 减到 0 的最小操作数

1. 题目链接:将 x 减到 0 的最小操作数 (medium)

2. 题目描述:

3. 解法(滑动窗口):

算法思路:

题目要求的是数组「左端+右端」两段连续的、和为x 的最短数组,信息量稍微多一些,不易理清

思路;我们可以转化成求数组内一段连续的、和为sum(nums) - x 的最长数组。此时,就是熟

悉的「滑动窗口」问题了。

算法流程:

a. 转化问题:求 target = sum(nums) - x 。如果 target < 0 ,问题无解;
b. 初始化左右指针 l = 0 , r = 0 (滑动窗口区间表示为 [l, r) ,左右区间是否开闭很重要,必须设定与代码一致),记录当前滑动窗口内数组和的变量 sum = 0 ,记录当前满足条件数组的最大区间长度 maxLen = -1 ;
c. 当r 小于等于数组长度时,一直循环:

i. 如果 sum < target ,右移右指针,直至变量和大于等于 target ,或右指针已经移到头;

ii. 如果 sum > target ,右移左指针,直至变量和小于等于 target ,或左指针已经移到头;

iii. 如果经过前两步的左右移动使得 sum == target ,维护满足条件数组的最大长度,并让 下个元素进入窗口;
d. 循环结束后,如果 maxLen 的值有意义,则计算结果返回;否则,返回-1 。

C++ 算法代码:

cpp 复制代码
class Solution {
public:
    int minOperations(vector<int>& nums, int x) {
        int total = accumulate(nums.begin(), nums.end(), 0);
        int target = total - x;
        if (target < 0) {
            return -1;
        }

        int left = 0;
        int right = 0;
        int len = -1;
        int sum = 0;

        int n = nums.size();

        for (right = 0; right < n; right++) {
            sum += nums[right];
            while (sum > target) {
                sum -= nums[left++];
            }
            if (sum == target)
                len = (right - left + 1) > len ? (right - left + 1) : len;
        }
        return len == -1 ? len : n - len;
    }
};

C++ 代码结果:

Java 算法代码:

java 复制代码
class Solution
{
    public int minOperations(int[] nums, int x)
    {
        int sum = 0;
        for(int a : nums) sum += a;
        int target = sum - x;
        // 处理细节
        if(target < 0) return -1;
        int ret = -1;
        for(int left = 0, right = 0, tmp = 0; right < nums.length; right++)
        {
            tmp += nums[right]; // 进窗口
            while(tmp > target) // 判断
            tmp -= nums[left++]; // 出窗口
            if(tmp == target)
                ret = Math.max(ret, right - left + 1); // 更新结果
        }
        if(ret == -1) return ret;
        else return nums.length - ret;
    }
}

Java 运行结果:

相关推荐
難釋懷14 小时前
Redis数据结构-Set结构
数据结构·redis·bootstrap
如何原谅奋力过但无声17 小时前
【灵神高频面试题合集06-08】反转链表、快慢指针(环形链表/重排链表)、前后指针(删除链表/链表去重)
数据结构·python·算法·leetcode·链表
平行侠17 小时前
037插入排序 - 整理扑克牌的算法
数据结构·算法
,,?!,19 小时前
数据结构算法-排序算法
数据结构·算法·排序算法
‎ദ്ദിᵔ.˛.ᵔ₎21 小时前
C++哈希表
数据结构·c++·散列表
阿旭超级学得完1 天前
C++11(初始化)
java·开发语言·数据结构·c++·算法
云淡风轻~窗明几净1 天前
关于角谷猜想的五行小猜想
数据结构·算法
Languorous.1 天前
C++数据结构进阶|并查集(Union-Find)详解:从原理到面试实战
数据结构·c++·面试
Languorous.1 天前
C++数据结构进阶|堆(Heap)详解:从手写实现到面试高频实战
数据结构·c++·面试
玛卡巴卡ldf1 天前
【LeetCode 手撕算法】(栈)有效括号、最小栈、字符串解码、每日温度、柱状图最大矩形
java·数据结构·算法·leetcode·力扣