题目:

给定一个整数数组 arr,找到 min(b) 的总和,其中 b 的范围为 arr 的每个(连续)子数组。

由于答案可能很大,因此 返回答案模 10^9 + 7 。

示例 1:

输入:arr = [3,1,2,4]
输出:17
解释:
子数组为 [3],[1],[2],[4],[3,1],[1,2],[2,4],[3,1,2],[1,2,4],[3,1,2,4]。
最小值为 3,1,2,4,1,1,2,1,1,1,和为 17。
示例 2:

输入:arr = [11,81,94,43,3]
输出:444

提示:

1 <= arr.length <= 3 * 104
1 <= arr[i] <= 3 * 104

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/sum-of-subarray-minimums
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

先记录一下我没通过的解题思路:

我想利用双循环来解题,外层循环表示从连续子数组的开始,内层循环表示连续子数组的结束,这样数组中的每个单个元素也能相加,遍历过程中也能更新最小值,以及得到求和,最后将和与10^9 + 7取模即可。

但是用例过不去,我还找不出来原因,哭了我这个小垃圾~

问题代码:

 1 class Solution {
2 public int sumSubarrayMins(int[] arr) {
3 int sum = 0, n = arr.length;
4 for(int i = 0; i < n; i++){
5 sum += arr[i];
6 int min_num = arr[i];
7 for(int j = i + 1; j < n; j++){
8 if(arr[j] > min_num){
9 sum += min_num;
10 }else{
11 sum += arr[j];
12 min_num = Math.min(min_num, arr[j]);
13 }
14 }
15 }
16 int MOD = 1_000_000_007;
17 return (int)(sum % MOD);
18 }
19 }

参考评论区的各位大佬,得到以下题解~

单调栈+贡献法:

①贡献法:

例如:[1,4,2,3,1]中2是子数组[2]、[4,2]、[2,3]、[4,2,3]的最小值,那么2对答案的贡献就是2*4 = 8。

由于以2为连续子数组中的最小值,那么连续子数组中就不能包含比2小的数,因此就可以找到2左右两侧比它小的数的下标,这样就可以确定以2为最小值的连续子数组的边界范围。左右两边比2小的边界范围在(0,4),闭区间为[1,3]。

设arr[i]对应的边界为开区间(L,R),子数组需要包含arr[i],故:

  • 左边的下标可以是:L, L+1, L+2,...,i,一共有i-L个;
  • 右边的下标可以是:i,i+1,i+2,...,R-1,一共有R-i个;

arr中不含重复元素的前提下,以arr[i]为最小值的数组个数为 (i-L)*(R-i) ,对答案的贡献值为 arr[i] *(i-L)*(R-i) (类比:上衣有n件,裤子有m条,则一共有 n * m中搭配方法)。注意:如果左侧没有比 arr[i]小的元素,则L= -1,如果右侧没有比 arr[i]小的元素,则R= n,

如果arr中包含重复元素,例如[1,2,4,2,3,1],第一个2和第二个2对应的边界为(0,5),都包含了子数组[2,4,2,3],这样就会计算同一个子数组。

这样需要修改一下边界的定义,将左边界改为小于等于arr[i]的元素下标,右边界依然为小于arr[i]的元素下标,那么:第一个2对应的边界为(0,5),第二个2对应的边界为(1,5)。

单调栈:参考@【爪哇缪斯L5】:https://leetcode.cn/problems/sum-of-subarray-minimums/solution/-by-muse-77-367z/

构建一个单调递增的栈,里面存放的是元素的下标,如果栈为空,那么直接入栈,如果栈不为空,则进行如下操作:

  • 如果 栈顶元素对应的值 >= arr[i]:就弹出栈顶元素,直到栈为空或者栈顶元素对应的值更小,此时栈顶就是最近更小值的位置,即为左边界,即(新的栈顶元素, i),反过来想想,对于每个出栈的元素,当前元素arr[i]就是向右比它更小的第一个元素,这样也就得到右边界(结合下面的图解进行理解);
  • 如果 栈顶元素对应的值 < arr[i]:则arr[i]直接入栈

例如:

当遍历完整个数组以后,栈中仍然还存在元素,让剩下元素出栈,根据题目中【提示】部分描述,1 <= arr[i] <= 3 * 10^4,所以,arr数组中所有元素都是大于0的,那么我们就虚拟一个元素将其放入堆栈中,因为堆栈中所有元素的值都大于0,所以也就都会被执行出栈操作了。

 java代码:

 1 class Solution {
2 private static final long MOD = 1000000007;
3 public int sumSubarrayMins(int[] arr) {
4 long ans = 0;
5 Deque<Integer> stack = new ArrayDeque<>();
6 //最左边界哨兵入栈,-1是索引
7 stack.push(-1);
8 for(int i = 0; i <= arr.length; i++){
9 //添加末尾哨兵,已经超过数组长度
10 //则设一个虚拟哨兵索引为n对应的值为0,方便剩余元素出栈
11 int m = i < arr.length ? arr[i] : 0;
12 while(stack.size() > 1 && arr[stack.peek()] >= m){
13 int r = stack.pop();
14 //当前栈顶的贡献值
15 //(栈顶-弹出后的栈顶)*(当前值的下标-弹出栈顶)
16 ans += (long) arr[r] * (r - stack.peek()) * (i - r);
17 }
18 stack.push(i);
19 }
20 return (int)(ans % MOD);
21 }
22 }

力扣907(java)-子数组的最小值之和(中等)的更多相关文章

  1. 【LeetCode】 907 子数组的最小值之和

    Decrisption Given an array of integers arr, find the sum of min(b), where b ranges over every (conti ...

  2. [Swift]LeetCode907. 子数组的最小值之和 | Sum of Subarray Minimums

    Given an array of integers A, find the sum of min(B), where B ranges over every (contiguous) subarra ...

  3. 力扣算法经典第一题——两数之和(Java两种方式实现)

    一.题目 难度:简单 给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数, 并返回它们的数组下标. 你可以假设每种输入只会对应一 ...

  4. 力扣(LeetCode)15. 三数之和

    给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?找出所有满足条件且不重复的三元组. 注意:答案中不可以包含重复的三元组. ...

  5. 力扣(LeetCode)1.两数之和

    给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标. 你可以假设每种输入只会对应一个答案.但是,你不能重复利用这个数组中同样的元 ...

  6. [LeetCode] 907. Sum of Subarray Minimums 子数组最小值之和

    Given an array of integers A, find the sum of min(B), where B ranges over every (contiguous) subarra ...

  7. 152.Maximum Product Subarray---dp---连续子数组的最大乘积---《编程之美》2.13子数组的最大乘积

    题目链接:https://leetcode.com/problems/maximum-product-subarray/description/ 题目大意:给出一串数组,找出连续子数组中乘积最大的子数 ...

  8. N元数组的子数组之和的最大值

    题目:有N个整数的元素的一维数组,求子数组中元素之和中最大的一组(思想:动态规划) 分析: 设该数组为array[N], 那么对于array[i]该不该在元素之和最大的那个子数组中呢?首先,不如假设a ...

  9. [LeetCode] Continuous Subarray Sum 连续的子数组之和

    Given a list of non-negative numbers and a target integer k, write a function to check if the array ...

  10. 累加和为 K 的子数组问题

    累加和为 K 的子数组问题 作者:Grey 原文地址: 博客园:累加和为 K 的子数组问题 CSDN:累加和为 K 的子数组问题 题目说明 数组全为正数,且每个数各不相同,求累加和为K的子数组组合有哪 ...

随机推荐

  1. 使用ScottPlot库在.NET WinForms中快速实现大型数据集的交互式显示

    前言 在.NET应用开发中数据集的交互式显示是一个非常常见的功能,如需要创建折线图.柱状图.饼图.散点图等不同类型的图表将数据呈现出来,帮助人们更好地理解数据.发现规律,并支持决策和沟通.本文我们将一 ...

  2. 回声消除(Acoustic Echo Cancellation)中遇到的几个常见问题思考

    什么才是好的回声消除效果   个人的理解:好的回声消除算法,要满足两个条件:一个是回声确实被消除了,另外一个是麦克风采集到音频信号不能被消除,常见的就是人的声音信号.这个算法只是提供了一种方法,具体的 ...

  3. maven解决尝试手段

    发现原来用的buildBody不好用,百度这个请求有异于其他sdk 关于写身份证接口那边:首先报有两个slf4j冲突,经过查阅,不能包含两个slf4j遂写了exclusion,但是排除不了,要使用** ...

  4. Linux 文件权限、VIM、防火墙

    Linux 文件权限.VIM.防火墙 目录 Linux 文件权限.VIM.防火墙 SSH连接 环境变量 权限 更改文件所属 更改文件权限 su和sudo 包管理器 VI/VIM iptables防火墙 ...

  5. 3DCAT实时云渲染助力广府庙会元宇宙焕新亮相,开启线上奇趣之旅!

    超 400 万人次打卡,商圈营业额逾 3.6 亿元,2023 年广府庙会于2023年2月11日圆满落幕. 活动期间,佳境美如画,融合VR.AR.虚拟直播等技术的广府庙会元宇宙焕新亮相,群众只需点击一个 ...

  6. 三维模型3DTile格式轻量化云端处理技术方法分析

    三维模型3DTile格式轻量化云端处理技术方法分析 在现代的地理信息系统 (GIS) 中,3D Tiles 是一种很重要的数据格式,用于存储和传输大规模地理空间数据.然而,由于其数据密度高,传输和加载 ...

  7. objective-c之Class底层结构探索

    isa 走位图 在讲 OC->Class 底层类结构之前,先看下下面这张图: 通过isa走位图 得出的结论是: 1,类,父类,元类都包含了 isa, superclass 2,对象isa指向类对 ...

  8. 记一次查询优化,mybatis查询oracle卡,直接拿sql数据库查询很快

    调整前 <select id="getList" resultMap="BaseResultMap" parameterType="java.u ...

  9. node14.20.0安装pnpm5.15.0兼容

    1,执行命令:npm install -g pnpm@5.15.0 2,设置淘宝镜像源: pnpm config set registry https://registry.npm.taobao.or ...

  10. Non-local Network:人类早期在CV驯服Transformer尝试 | CVPR 2018

    Non-local操作是早期self-attention在视觉任务上的尝试,核心在于依照相似度加权其它特征对当前特征进行增强,实现方式十分简洁,为后续的很多相关研究提供了参考   来源:晓飞的算法工程 ...