算法38:子数组的最小值之和(力扣907题)----单调栈

2024-01-28 18:52

本文主要是介绍算法38:子数组的最小值之和(力扣907题)----单调栈,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目:

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

示例 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. 假设数组为1,那么子数组最小值就为1.

2. 假设数组为{1,2} 那么子数组就可以为{1} 和 {1,2}

3. 假设数组为{1,2,3} 那么子数组就可以为{1}、{1,2} 和 {1,2,3}

总结:如果以1为子数组最小值,那么1后面出现的比1大的数,有几个数,子数组就为 n + 1。

数组{1,2,3}中,以1为最小值,那么1后面有2个数比1大,子数组就为 2 + 1 = 3个。

假设数组为{1,2,3}, 现在在最小值1前面加一个数2, 变成{2,1,2,3}.  那么子数组就为

{2,1}、{2,1,2} 、{2,1,2,3}

{1}、{1、2}、{1,2,3}

我们发现,子数组数量是原始数组{1,2,3}的2倍,即 2 * 3 = 6 个。

假设,我们在现有的数组中,前面在添加一个元素3. 变成{3,2,1,2,3}.  那么子数组为:

{3,2,1} 、{3,2,1,2} 、{3,2,1,3}

{2,1}、{2,1,2} 、{2,1,2,3}

{1}、{1、2}、{1,2,3}

我们发现,子数组数量是原始数组{1,2,3}的3倍,即 3 * 3 = 9 个.

总结:如果以1为子数组最小值,那么1前面出现的比1大的数,有几个,那么就是 (N +1)* 原始个数。

 以{3,2,1,2,3}为例子。

如果以1为最小值,1后面有2个数比1大,得到 2+1 = 3; 1前面有2个比1大,得到2+1= 3; 3*3 =9; 如果以1为最小值,那么一共有9个子数组。

如果以下标为1的2为最小值,可得 (1+1)* (0+1) = 2; 即如果以下标为1的2值为最小值,子数组有2个。即 {3,2} 和 {2}

如果以下标为3的位置的2值为最小值,前一个位置为1,即0个。前方0个比自己大的;后面1个3比自己大,可得 (0+1)*(1+1) = 2个;即{2}和{2、3}

依次类推,可以得到全部结果.

package code04.单调栈_01;import java.util.Stack;/*** 力扣力扣907题: 子数组的最小值*  https://leetcode.com/problems/sum-of-subarray-minimums/*/
public class Code04_SumOfMinValueInArray {public int sumSubarrayMins(int[] arr){int[][] dp = dp(arr);//long比int能存更长的数据long ans = 0;for (int i = 0; i < arr.length; i++) {//当前数int cur = arr[i];//左侧小于等于当前数的个数int left = i - dp[i][0];//右侧小于等于当前数的个数int right = dp[i][1] - i;ans +=  left * right * (long)cur;ans %= 1000000007;}return (int) ans;}//单调栈,统计出每个位置左、右侧比当前数小的位置public int[][] dp(int[] arr){if(arr == null || arr.length == 0) {return null;}//当前业务需要统计出2列信息, 即左侧比自己小的位置,右侧比自己小的位置int[][] dp = new int[arr.length][2];Stack<Integer> stack = new Stack<>();for (int i = 0; i < arr.length; i++){while (!stack.isEmpty() && arr[stack.peek()] > arr[i]){int cur = stack.pop();// -1代表不存在左侧比cur下标对应的值更小的值int leftIndex = stack.isEmpty() ? -1 : stack.peek();dp[cur][0] = leftIndex;dp[cur][1] = i;}//放入下标stack.push(i);}int rightIndex = arr.length;//栈中剩余元素,保持单调增while (!stack.isEmpty()) {int cur = stack.pop();// -1代表不存在左侧比cur下标对应的值更小的值int leftIndex = stack.isEmpty() ? -1 : stack.peek();dp[cur][0] = leftIndex;//因为单调增、所有右侧不存在比自己还小的值了dp[cur][1] = rightIndex;}return dp;}public static void main(String[] args) {Code04_SumOfMinValueInArray ss = new Code04_SumOfMinValueInArray();int[] aa = {3,1,2,4};System.out.println(ss.sumSubarrayMins(aa));}
}

虽然测试通过了,但是执行用时99ms,只击败了17%的用户,说明代码不够优秀。

技巧就是,将java原有的Stack替换成自己实现的数组。因为自己实现的数组是固定的,而Stack是需要不断经过扩容的。这样优化,效果很明显。

package code04.单调栈_01;import java.util.Stack;/*** 力扣力扣907题: 子数组的最小值*  https://leetcode.com/problems/sum-of-subarray-minimums/*/
public class Code04_SumOfMinValueInArray_opt {public int sumSubarrayMins(int[] arr){int[][] dp = dp(arr);//long比int能存更长的数据long ans = 0;for (int i = 0; i < arr.length; i++) {//当前数int cur = arr[i];//左侧小于等于当前数的个数int left = i - dp[i][0];//右侧小于等于当前数的个数int right = dp[i][1] - i;ans +=  left * right * (long)cur;ans %= 1000000007;}return (int) ans;}//单调栈,统计出每个位置左、右侧比当前数小的位置public int[][] dp(int[] arr){if(arr == null || arr.length == 0) {return null;}//当前业务需要统计出2列信息, 即左侧比自己小的位置,右侧比自己小的位置int[][] dp = new int[arr.length][2];int[] stack = new int[arr.length];int stackSize = 0;for (int i = 0; i < arr.length; i++){while (stackSize != 0 && arr[stack[stackSize-1]] > arr[i]){int cur = stack[--stackSize];// -1代表不存在左侧比cur下标对应的值更小的值int leftIndex = stackSize == 0 ? -1 : stack[stackSize - 1];dp[cur][0] = leftIndex;dp[cur][1] = i;}//放入下标stack[stackSize++] = i;}int rightIndex = arr.length;//栈中剩余元素,保持单调增while (stackSize != 0) {int cur = stack[--stackSize];// -1代表不存在左侧比cur下标对应的值更小的值int leftIndex = stackSize == 0 ? -1 : stack[stackSize - 1];dp[cur][0] = leftIndex;//因为单调增、所有右侧不存在比自己还小的值了dp[cur][1] = rightIndex;}return dp;}public static void main(String[] args) {Code04_SumOfMinValueInArray_opt ss = new Code04_SumOfMinValueInArray_opt();int[] aa = {3,1,2,4};int[][] dp = ss.dp(aa);for (int i = 0; i < dp.length; i++) {System.out.println("当前下标 :" + i + ", 左侧小值: " + dp[i][0] + ", 右侧小值: " + dp[i][1]);}System.out.println(ss.sumSubarrayMins(aa));}
}

优化完以后,效果很明显: 

这篇关于算法38:子数组的最小值之和(力扣907题)----单调栈的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/654494

相关文章

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

Springboot实现推荐系统的协同过滤算法

《Springboot实现推荐系统的协同过滤算法》协同过滤算法是一种在推荐系统中广泛使用的算法,用于预测用户对物品(如商品、电影、音乐等)的偏好,从而实现个性化推荐,下面给大家介绍Springboot... 目录前言基本原理 算法分类 计算方法应用场景 代码实现 前言协同过滤算法(Collaborativ

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的