算法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));
    }
}

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/354240.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java集合总览

1.总览 Java中的集合分List、Set、Queue、Map 4种类型。 List&#xff1a;大多数实现元素可以为null&#xff0c;可重复&#xff0c;底层是数组或链表的结构&#xff0c;支持动态扩容 Set&#xff1a;大多数实现元素可以为null但只能是1个&#xff0c;不能重复&#xff0c; …

MySQL 聚集与非聚集索引

文章目录 1.聚集索引1.1 介绍1.2 优点1.3 缺点 2.非聚集索引3.区别参考文献 MySQL 中&#xff0c;根据索引树叶结点存放数据行还是数据行的地址&#xff0c;可以将索引分为两类&#xff1a; 存放数据行&#xff1a;聚集索引存放数据行地址&#xff1a;非聚集索引 InnoDB 使用聚…

Linux学习之文件系统与动静态库

目录 一&#xff0c;文件的管理 什么是磁盘&#xff1f; 磁盘的逻辑抽象结构 格式化 inode 挂载 软硬链接 二&#xff0c;动静态库 什么是动静态库&#xff1f; 1.站在库的制作者角度 静态库&#xff1a; 制作一个静态库 2.站在静态库使用者的角度 动态库 作为制…

windows安装PostgreSQL后进行远程连接,发生SSL错误

1. 报错情况 SSL 关闭 的 pg_hba.conf 记录 (pgjdbc: autodetected server-encoding to be GB2312, if the message is not readable, please check database logs and/or host, port, dbname, user, password, pg_hba.conf) 或是乱码提示&#xff0c;提示中有SSL、 pg_hba.con…

C语言KR圣经笔记 5.12 复杂声明

5.12 复杂声明 C 语言有时会因为声明的语法而受到谴责&#xff0c;特别是涉及函数指针的声明语法。语法试图使声明和使用一致&#xff1b;在简单的情况下它的效果不错&#xff0c;但在更复杂的情况下会让人困惑&#xff0c;因为声明不能从左往右读&#xff0c;而且括号被过度使…

RabbitMQ快速上手

首先他的需求实在什么地方。我美哟明显的感受到。 它给我的最大感受就是脱裤子放屁——多此一举&#xff0c;的感觉。 他将信息发送给服务端中间件。在由MQ服务器发送消息。 服务器会监听消息。 但是它不仅仅局限于削峰填谷和稳定发送信息的功能&#xff0c;它还有其他重要…

Nacos(先解释专属名词,然后大白话讲解+安装配置教程+代码实例 信我,看了必会!!点进来吧各位大佬们)

Nacos 先来一个注意&#xff1a;以下涉及到配置和项目的创建&#xff0c;我的jdk是jdk17&#xff0c;idea是2023.2.4&#xff0c;并且是社区版的idea 1.什么是Nacos Nacos是Dyamic Naming and Configuration Service的首字母简称&#xff0c;一个更易于构建云原生应用的动态…

重生奇迹MU平民玩家推荐的职业

女魔法师 女魔法师是一个非常适合平民玩家的职业选择。她拥有着强大的魔法攻击能力&#xff0c;可以轻松地击败敌人。而且女魔法师的装备价格相对较低&#xff0c;适合玩家们的经济实力。 精灵射手 精灵射手是一个非常灵活的职业选择。他们可以远程攻击&#xff0c;可以在战…

数据结构之生成树及最小生成树

数据结构之生成树及最小生成树 1、生成树概念2、最小生成树 数据结构是程序设计的重要基础&#xff0c;它所讨论的内容和技术对从事软件项目的开发有重要作用。学习数据结构要达到的目标是学会从问题出发&#xff0c;分析和研究计算机加工的数据的特性&#xff0c;以便为应用所…

centos7 挂载windows共享文件夹报错提示写保护

centos7挂载windows共享时&#xff0c;提示被共享的位置写保护&#xff0c;只能以只读方式挂载&#xff0c;紧接着就是以只读方式挂载失败 原因是组件少装了 yum install cifs-utils 安装完后&#xff0c;正常挂载使用。 下载离线安装包 下载离线包下载工具 下载离线安装包…

神经网络:表述(Neural Networks: Representation)

1.非线性假设 无论是线性回归还是逻辑回归&#xff0c;当特征太多时&#xff0c;计算的负荷会非常大。 案例&#xff1a; 假设我们有非常多的特征&#xff0c;例如大于 100 个变量&#xff0c;我们希望用这 100 个特征来构建一个非线性的多项式模型&#xff0c;结果将是数量非…

在windows环境下安装hadoop

Hadoop是一个分布式系统基础架构。用户可以在不了解分布式底层细节的情况下&#xff0c;开发分布式程序。但这个架构是基于java语言开发的&#xff0c;所以要先进行jdk的安装&#xff0c;如果电脑已经配置过jdk或者是曾经运行成功过java文件&#xff0c;那就可以跳过第一步。 …

可解释性人工智能(XAI)概述

文章目录 每日一句正能量前言可解释性人工智能&#xff08;XAI&#xff09;定义研究的作用应用领域XAI的目标后记 每日一句正能量 一个人若想拥有聪明才智&#xff0c;便需要不断地学习积累。 前言 人工智能&#xff08;AI&#xff09;的发展速度迅猛&#xff0c;并在许多领域…

消息中间件之八股面试回答篇:三、RabbitMQ如何解决消息堆积问题(100万条消息堆积)+RabbitMQ高可用性和强一致性机制+回答模板

RabbitMQ中的消息堆积问题 当生产者发送消息的速度超过了消费者处理消息的速度&#xff0c;就会导致队列中的消息堆积&#xff0c;直到队列存储消息达到上限。之后发送的消息就会成为死信&#xff0c;可能会被丢弃&#xff0c;这就是消息堆积问题。 解决消息堆积有三种种思路…

【学网攻】 第(13)节 -- 动态路由(OSPF)

系列文章目录 目录 系列文章目录 文章目录 前言 一、动态路由是什么&#xff1f; 二、实验 1.引入 实验拓扑图 实验配置 实验验证 总结 文章目录 【学网攻】 第(1)节 -- 认识网络【学网攻】 第(2)节 -- 交换机认识及使用【学网攻】 第(3)节 -- 交换机配置聚合端口【学…

什么是数据库的三级模式两级映象?

三级模式两级映象结构图 概念 三级模式 内模式&#xff1a;也称为存储模式&#xff0c;是数据物理结构和存储方式的描述&#xff0c;是数据在数据库内部的表示方式。定义所有的内部记录类型、索引和文件组织方式&#xff0c;以及数据控制方面的细节。模式&#xff1a;又称概念…

Ubuntu20.04安装Google浏览器

一.在 Ubuntu 上安装 Google Chrome Chrome 不是一个开源的浏览器&#xff0c;并且它不被包含在标准的 Ubuntu 软件源中。在 Ubuntu 中安装 Google Chrome 是一个非常直接的过程。我们将会从官方网站下载安装文件&#xff0c;并且通过命令行工具来安装它。 1.1 下载 Google Ch…

Cesium材质特效

文章目录 0.引言1.视频材质2.分辨率尺度3.云4.雾5.动态水面6.雷达扫描7.流动线8.电子围栏9.粒子烟花10.粒子火焰11.粒子天气 0.引言 现有的gis开发方向较流行的是webgis开发&#xff0c;其中Cesium是一款开源的WebGIS库&#xff0c;主要用于实时地球和空间数据的可视化和分析。…

函数式接口当参数使用

如果函数式接口作为一个方法的参数&#xff0c;就以为着要方法调用方自己实现业务逻辑&#xff0c;常见的使用场景是一个业务整体逻辑是不相上下的&#xff0c;但是在某一个步骤有不同的逻辑&#xff0c;例如数据处理有不同的策略。上代码 package com.dj.lambda;import java.…

加密域可逆信息隐藏算法分类及评价指标

一、加密域可逆信息隐藏算法框架分类 加密图像可逆信息隐藏(RDHEI)是将图像加密和信息隐藏结合使用的一种技术。图像拥有者先对原始图像使用加密密钥keyc进行加密&#xff0c;信息隐藏者根据隐藏密钥keyd将秘密信息嵌入到密文图像中去。在接收端&#xff0c;接收者根据密钥key…