leetcode907. 子数组的最小值之和,单调栈

leetcode907. 子数组的最小值之和,单调栈

给定一个整数数组 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

目录

题目分析

这是一个关于子数组求和的问题。题目要求实现一个函数sumSubarrayMins,该函数接受一个整数数组arr,并返回数组中所有子数组的和的最小值。

算法介绍

为了找到子数组的和的最小值,我们可以使用一个栈来维护数组中每个元素的位置。通过维护每个元素左侧和右侧小于或等于该元素的最近位置,我们可以计算每个子数组的和,并找出所有子数组和的最小值。

算法步骤

  1. 初始化两个数组:left(左侧小于等于当前元素的最近位置)和right(右侧小于等于当前元素的最近位置)。
  2. 从左到右遍历数组,使用栈来维护每个元素的位置。
  3. 从右到左再次遍历数组,同样使用栈来维护每个元素的位置。
  4. 计算每个子数组的和,并累加到最终答案中。
  5. 返回答案,对MOD取模。

算法流程

开始 初始化 left, right, MOD 从左到右遍历 arr 维护栈 从右到左再次遍历 arr 维护栈 计算子数组和 累加到答案 返回答案 % MOD 结束

算法代码

cpp 复制代码
class Solution {
    const int MOD = 1e9 + 7;
public:
    int sumSubarrayMins(vector<int> &arr) {
        int n = arr.size();
    // 左边界 left[i] 为左侧严格小于 arr[i] 的最近元素位置(不存在时为 -1)
        vector<int> left(n, -1);
        stack<int> st;
        for (int i = 0; i < n; ++i) {
            while (!st.empty() && arr[st.top()] >= arr[i])
                st.pop(); // 移除无用数据
            if (!st.empty()) left[i] = st.top();
            st.push(i);
        }

    // 右边界 right[i] 为右侧小于等于 arr[i] 的最近元素位置(不存在时为 n)
        vector<int> right(n, n);
        while (!st.empty()) st.pop();
        for (int i = n - 1; i >= 0; --i) {
            while (!st.empty() && arr[st.top()] > arr[i])
                st.pop(); // 移除无用数据
            if (!st.empty()) right[i] = st.top();
            st.push(i);
        }

        long ans = 0L;
        for (int i = 0; i < n; ++i)
            ans += (long) arr[i] * (i - left[i]) * (right[i] - i); 
            // 累加贡献
        return ans % MOD;
    }
};

算法分析

  • 时间复杂度 :O(n),其中n是数组arr的长度。我们只需要遍历数组两次。
  • 空间复杂度:O(n),因为使用了额外的数组和栈空间。
  • 易错点
    • 确保正确地维护leftright数组。
    • 在计算子数组和时,确保正确处理边界条件。

相似题目

题目 链接
子数组求和的最小值 LeetCode 1036
最小子数组和 LeetCode 53

请注意,以上表格仅为示例,实际链接可能需要根据具体平台和题目编号进行调整。

相关推荐
闻缺陷则喜何志丹几秒前
【C++动态规划 图论】3243. 新增道路查询后的最短距离 I|1567
c++·算法·动态规划·力扣·图论·最短路·路径
charlie11451419112 分钟前
C++ STL CookBook
开发语言·c++·stl·c++20
Lenyiin19 分钟前
01.02、判定是否互为字符重排
算法·leetcode
小林熬夜学编程23 分钟前
【Linux网络编程】第十四弹---构建功能丰富的HTTP服务器:从状态码处理到服务函数扩展
linux·运维·服务器·c语言·网络·c++·http
倔强的石头10634 分钟前
【C++指南】类和对象(九):内部类
开发语言·c++
鸽鸽程序猿34 分钟前
【算法】【优选算法】宽搜(BFS)中队列的使用
算法·宽度优先·队列
Jackey_Song_Odd35 分钟前
C语言 单向链表反转问题
c语言·数据结构·算法·链表
Watermelo61738 分钟前
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
开发语言·前端·javascript·算法·数据挖掘·数据分析·ecmascript
乐之者v44 分钟前
leetCode43.字符串相乘
java·数据结构·算法
A懿轩A2 小时前
C/C++ 数据结构与算法【数组】 数组详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·数组