[LeeCode 862. 和至少为 K 的最短子数组]单调栈

[LeeCode 862. 和至少为 K 的最短子数组]单调栈

1. 题目链接

[LeeCode 862. 和至少为 K 的最短子数组]

2. 题意描述

题意描述

3. 解题思路

首先,预处理出数组的前缀和 pre p r e , 当区间 [l,r] [ l , r ] 的子数组和至少为 K K 时,那么有: pre[r]pre[l1]+K;
然后,就是需要求出以第 i i 个元素为左端点的子数组中,子数组和至少为K的最小长度,换句话说, 求出满足下面3个条件的右端点:

  1. ix i ≤ x
  2. pre[x]pre[i1]+K p r e [ x ] ≥ p r e [ i − 1 ] + K
  3. min{xl+1} m i n { x − l + 1 }

可以发现, x x 越靠小,且pre[x]越大越好. 反向遍历就可以保证 x x 递减, 然后维护一个单调递减的栈stk,保证添加进来的 x x 满足:pre[x]<pre[stk.top()].此时,对单调栈进行二分查找,就可以求出子数组的右端点了.
算法总复杂度是: O(nlog2n) O ( n l o g 2 n )

4. 参考代码

#ifdef __LOCAL_WONZY__
#include <bits/stdc++.h>
using namespace std;
#endif

class Solution {
public:
    typedef long long ll;
    static const int inf = 0x3f3f3f3f;
    vector<long long> pre;
    long long sum(int le, int ri) {
        return le == 0 ? pre[ri] : pre[ri] - pre[le - 1];
    }
    int shortestSubarray(vector<int>& A, int K) {
        int n = A.size();
        pre = vector<long long>(n);
        pre[0] = A[0];
        for(int i = 1; i < n; ++i) pre[i] = pre[i - 1] + A[i];
        vector<int> stk(n);
        int sz = 0, len = inf;
        for(int i = n - 1; i >= 0; --i) {
            while(sz > 0 && pre[stk[sz - 1]] <= pre[i]) -- sz;
            stk[sz ++] = i;
            long long val = (i == 0 ? 0 : pre[i - 1]) + K;
            int lb = 0, ub = sz - 1, md, j = inf;
            while(lb <= ub) {
                int md = (lb + ub) >> 1;
                if(pre[stk[md]] >= val) j = stk[md], lb = md + 1;
                else ub = md - 1;
            }
            len = min(len, j - i + 1);
        }
        return len == inf ? -1 : len;
    }
};

#ifdef __LOCAL_WONZY__
int main() {
#ifdef __LOCAL_WONZY__
    freopen("input-1.txt", "r", stdin);
#endif
    vector<int> a = {77,19,35,10,-14};
    Solution s;
    cout << s.shortestSubarray(a, 19) << endl;;
    return 0;
}
#endif
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
LeetCode是一个在线的编程题库,提供了各种各样的算法和数据结构问题供开发者练习和提高编程能力。其中,提到了三个与数组相关的问题。 第一个引用是关于合并两个有序数组的问题,即将两个有序的整数数组合并成一个有序数组。具体的解法可以使用双指针法,分别指向两个数组的末尾,然后比较两个指针指向的元素大小,将较大的元素放入结果数组的末尾,然后将指针向前移动。这个过程重复直到其中一个数组遍历完毕,然后将剩下的元素依次放入结果数组中。 第二个引用是关于移除元素的问题,即移除数组中指定的元素,并返回新数组的长度。可以使用双指针法,左指针维护非指定元素的末尾位置,右指针遍历数组。当右指针指向的元素不等于指定元素时,将右指针指向的元素赋值给左指针指向的位置,然后将左指针和右指针都向前移动一位。直到右指针遍历完整个数组,最后返回左指针的值,即为新数组的长度。 第三个引用也是关于移动零元素的问题,即将数组中的零元素移动到数组末尾,同时保持非零元素的相对顺序不变。可以使用双指针法,左指针维护非零元素的末尾位置,右指针遍历数组。当右指针指向的元素不等于零时,将右指针指向的元素与左指针指向的元素交换位置,然后将左指针和右指针都向前移动一位。直到右指针遍历完整个数组,即完成了零元素的移动。 这些问题都可以使用双指针法解决,利用双指针在数组上进行遍历和操作,实现对数组的操作和处理。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [LeeCode每日一题–合并两个有序数组](https://download.csdn.net/download/weixin_38645669/14856034)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [LeeCode 数组题目](https://blog.csdn.net/weixin_43763903/article/details/114675786)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值