首页 > 其他分享 >862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组

862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组

时间:2022-10-29 23:00:37浏览次数:53  
标签:nums int 862 list 短子 数组 ans 树状


题目描述

这是 LeetCode 上的 ​​863. 二叉树中所有距离为 K 的结点​​ ,难度为 困难

Tag : 「前缀和」、「离散化」、「二分」、「树状数组」

给你一个整数数组 ​​nums​​​ 和一个整数 ​​k​​​ ,找出 ​​nums​​​ 中和至少为 ​​k​​​ 的 最短非空子数组 ,并返回该子数组的长度。如果不存在这样的 子数组 ,返回 ​​-1​​ 。

子数组 是数组中 连续 的一部分。

示例 1:

输入:nums = [1], k = 1

输出:1

示例 2:

输入:nums = [1,2], k = 4

输出:-1

示例 3:

输入:nums = [2,-1,2], k = 3

输出:3

提示:

  • 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_掘金·日新计划
  • 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_后端_02
  • 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_03

前缀和 + 离散化 + 权值树状数组

由于求解的对象是子数组,容易联想到求连续段之和,容易联想到「前缀和」,假设我们预处理出的前缀和数组为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_04(为了方便,我们令前缀和数组坐标从 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_掘金·日新计划_05

即每个 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_06 而言,本质上是找满足「862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_07」条件的最大下标 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_后端_08,其中 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_后端_08 的取值范围为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_10,从而知道以 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_掘金·日新计划_11 作为右端点时,满足条件的最短子数组长度为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_Java_12

先考虑存在负数域的问题,由于我们需要使用 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_13,以及对应的 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_14,同时 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_15 的取值为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_16(过大),我们可以通过「离散化」手段将其映射到 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_Java_17 倍的数组长度,即大小为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_掘金·日新计划_18

随后来考虑如何求解「满足条件的最大下标」问题,可以通过「权值树状数组」来做:对于每个 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_掘金·日新计划_19 而言,我们利用「权值树状数组」来维护满足大于等于 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_后端_20 的最大下标。起始我们先初始化树状数组为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_后端_21,遍历过程中,查询是否存在满足条件的下标(若不为 ​​​-1​​​ 则更新 ​​ans​​),并更新权值树状数组对应的最大下标即可。

代码:

class Solution {
static int N = 200010;
static int[] tr = new int[N], sum = new int[N];
int n, m, ans;
int lowbit(int x) {
return x & -x;
}
void update(int val, int loc) {
for (int i = val; i < m; i += lowbit(i)) tr[i] = Math.max(tr[i], loc);
}
int query(int x) {
int ans = -1;
for (int i = x; i > 0; i -= lowbit(i)) ans = Math.max(ans, tr[i]);
return ans;
}
int getIdx(List<Long> list, long x) {
int l = 0, r = list.size() - 1;
while (l < r) {
int mid = l + r >> 1;
if (list.get(mid) >= x) r = mid;
else l = mid + 1;
}
return r + 1;
}
public int shortestSubarray(int[] nums, int k) {
n = nums.length; m = 2 * n + 10; ans = n + 10;
Arrays.fill(tr, -1);
long[] temp = new long[m];
List<Long> list = new ArrayList<>();
list.add(0L);
for (int i = 1; i <= 2 * n + 1; i++) {
if (i <= n) temp[i] = temp[i - 1] + nums[i - 1];
else temp[i] = temp[i - (n + 1)] + k;
list.add(temp[i]);
}
Collections.sort(list);
for (int i = 0; i <= 2 * n + 1; i++) sum[i] = getIdx(list, temp[i]);
update(sum[n + 1], 0);
for (int i = 1; i <= n; i++) {
int j = query(sum[i]);
if (j != -1) ans = Math.min(ans, i - j);
update(sum[n + 1 + i], i);
}
return ans == n + 10 ? -1 : ans;
}
}
  • 时间复杂度:预处理前缀和的的复杂度为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_22,排序并进行离散化的复杂度为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_23;构造答案的复杂度为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_23。整体复杂度为 862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_前缀和_23
  • 空间复杂度:862. 和至少为 K 的最短子数组 : 前缀和 + 离散化 + 权值树状数组_树状数组_22

最后

这是我们「刷穿 LeetCode」系列文章的第 ​​No.862​​ 篇,系列开始于 2021/01/01,截止于起始日 LeetCode 上共有 1916 道题目,部分是有锁题,我们将先把所有不带锁的题目刷完。

在这个系列文章里面,除了讲解解题思路以外,还会尽可能给出最为简洁的代码。如果涉及通解还会相应的代码模板。

为了方便各位同学能够电脑上进行调试和提交代码,我建立了相关的仓库:​​github.com/SharingSour…​​ 。



标签:nums,int,862,list,短子,数组,ans,树状
From: https://blog.51cto.com/acoier/5806595

相关文章

  • 915. 分割数组 : 简单模拟题
    题目描述这是LeetCode上的​​915.分割数组​​,难度为中等。Tag:「模拟」给定一个数组 ​​nums​​​,将其划分为两个连续子数组 ​​left​​​ 和 ​​right......
  • C语言学习--结构体数组
      #include<stdio.h>#include<stdlib.h>structstu{intid;intage;charname[128];};intmain(void){//结构体数组:数组中的每一......
  • 动态规划树状数组优化(M元上升子序列)
    我们先来看简单版:P1637三元上升子序列这道题显然考虑dp,转移式子也很好写设f[i][j]表示以a[j]结尾长度为i的上升子序列个数。显然答案就是\(\sum\limits_{k=1}^{n}f_{3......
  • 60-ES10-数组方法扩展-flat与flatMap
     ......
  • 数组常用方法
    一、数组常用方法1.unshift(),从首位添加数据至原数组中,返回新数组的长度2.push(),从末位添加数据至原数组中,返回新数组的长度3.shift(),去掉数组的第一个......
  • 7种数组去重方法
    一、设定原数组 constarr=[22,22,'ture','ture',false,false,undefined,undefined,null,null,NaN,NaN,'NaN',0,0,'a','a',15,15,true,true,{},......
  • Shell脚本之数组
    概念数组(Array)是有序的元素序列。若将有限个类型相同的变量的集合命名,那么这个名称为数组名。组成数组的各个变量称为数组的分量,也称为数组的元素,有时也称为下标变量。......
  • js 字符串中包含逗号和分号分析成数组
    varstr="117.39755436808615,34.59211450864094;117.39783481906638,34.59185738594207;117.39825396841732,34.59151467824745;117.39895365857903,34.5909999082......
  • Shell脚本之数组排序
    数组排序(使用tr、sort、for)操作步骤;使用tr命令将数组内每个元素之间的空格替换为换行符;之后使用sort命令按从小到大重新排序;最后使用for循环遍历排序后的元素值。......
  • js 数组-过滤数据
    js数组-过滤数据filter()方法创建一个新的数组,新数组中的元素是通过检查指定数组中符合条件的所有元素。注意:filter()不会对空数组进行检测。注意:filter()不会改......