mirror of
https://github.com/halfrost/LeetCode-Go.git
synced 2025-07-05 08:27:30 +08:00
规范格式
This commit is contained in:
@ -0,0 +1,70 @@
|
||||
package leetcode
|
||||
|
||||
// 解法一 最快的解是 DP + 单调栈
|
||||
func sumSubarrayMins(A []int) int {
|
||||
stack, dp, res, mod := []int{}, make([]int, len(A)+1), 0, 1000000007
|
||||
stack = append(stack, -1)
|
||||
|
||||
for i := 0; i < len(A); i++ {
|
||||
for stack[len(stack)-1] != -1 && A[i] <= A[stack[len(stack)-1]] {
|
||||
stack = stack[:len(stack)-1]
|
||||
}
|
||||
dp[i+1] = (dp[stack[len(stack)-1]+1] + (i-stack[len(stack)-1])*A[i]) % mod
|
||||
stack = append(stack, i)
|
||||
res += dp[i+1]
|
||||
res %= mod
|
||||
}
|
||||
return res
|
||||
}
|
||||
|
||||
type pair struct {
|
||||
val int
|
||||
count int
|
||||
}
|
||||
|
||||
// 解法二 用两个单调栈
|
||||
func sumSubarrayMins1(A []int) int {
|
||||
res, n, mod := 0, len(A), 1000000007
|
||||
lefts, rights, leftStack, rightStack := make([]int, n), make([]int, n), []*pair{}, []*pair{}
|
||||
for i := 0; i < n; i++ {
|
||||
count := 1
|
||||
for len(leftStack) != 0 && leftStack[len(leftStack)-1].val > A[i] {
|
||||
count += leftStack[len(leftStack)-1].count
|
||||
leftStack = leftStack[:len(leftStack)-1]
|
||||
}
|
||||
leftStack = append(leftStack, &pair{val: A[i], count: count})
|
||||
lefts[i] = count
|
||||
}
|
||||
|
||||
for i := n - 1; i >= 0; i-- {
|
||||
count := 1
|
||||
for len(rightStack) != 0 && rightStack[len(rightStack)-1].val >= A[i] {
|
||||
count += rightStack[len(rightStack)-1].count
|
||||
rightStack = rightStack[:len(rightStack)-1]
|
||||
}
|
||||
rightStack = append(rightStack, &pair{val: A[i], count: count})
|
||||
rights[i] = count
|
||||
}
|
||||
|
||||
for i := 0; i < n; i++ {
|
||||
res = (res + A[i]*lefts[i]*rights[i]) % mod
|
||||
}
|
||||
return res
|
||||
}
|
||||
|
||||
// 解法三 暴力解法,中间很多重复判断子数组的情况
|
||||
func sumSubarrayMins2(A []int) int {
|
||||
res, mod := 0, 1000000007
|
||||
for i := 0; i < len(A); i++ {
|
||||
stack := []int{}
|
||||
stack = append(stack, A[i])
|
||||
for j := i; j < len(A); j++ {
|
||||
if stack[len(stack)-1] >= A[j] {
|
||||
stack = stack[:len(stack)-1]
|
||||
stack = append(stack, A[j])
|
||||
}
|
||||
res += stack[len(stack)-1]
|
||||
}
|
||||
}
|
||||
return res % mod
|
||||
}
|
@ -0,0 +1,42 @@
|
||||
package leetcode
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"testing"
|
||||
)
|
||||
|
||||
type question907 struct {
|
||||
para907
|
||||
ans907
|
||||
}
|
||||
|
||||
// para 是参数
|
||||
// one 代表第一个参数
|
||||
type para907 struct {
|
||||
one []int
|
||||
}
|
||||
|
||||
// ans 是答案
|
||||
// one 代表第一个答案
|
||||
type ans907 struct {
|
||||
one int
|
||||
}
|
||||
|
||||
func Test_Problem907(t *testing.T) {
|
||||
|
||||
qs := []question907{
|
||||
|
||||
question907{
|
||||
para907{[]int{3, 1, 2, 4}},
|
||||
ans907{17},
|
||||
},
|
||||
}
|
||||
|
||||
fmt.Printf("------------------------Leetcode Problem 907------------------------\n")
|
||||
|
||||
for _, q := range qs {
|
||||
_, p := q.ans907, q.para907
|
||||
fmt.Printf("【input】:%v 【output】:%v\n", p, sumSubarrayMins(p.one))
|
||||
}
|
||||
fmt.Printf("\n\n\n")
|
||||
}
|
45
leetcode/0907.Sum-of-Subarray-Minimums/README.md
Normal file
45
leetcode/0907.Sum-of-Subarray-Minimums/README.md
Normal file
@ -0,0 +1,45 @@
|
||||
# [907. Sum of Subarray Minimums](https://leetcode.com/problems/sum-of-subarray-minimums/)
|
||||
|
||||
## 题目
|
||||
|
||||
Given an array of integers A, find the sum of min(B), where B ranges over every (contiguous) subarray of A.
|
||||
|
||||
Since the answer may be large, return the answer modulo 10^9 + 7.
|
||||
|
||||
|
||||
|
||||
Example 1:
|
||||
|
||||
```c
|
||||
Input: [3,1,2,4]
|
||||
Output: 17
|
||||
Explanation: Subarrays are [3], [1], [2], [4], [3,1], [1,2], [2,4], [3,1,2], [1,2,4], [3,1,2,4].
|
||||
Minimums are 3, 1, 2, 4, 1, 1, 2, 1, 1, 1. Sum is 17.
|
||||
```
|
||||
|
||||
Note:
|
||||
|
||||
1. 1 <= A.length <= 30000
|
||||
2. 1 <= A[i] <= 30000
|
||||
|
||||
|
||||
## 题目大意
|
||||
|
||||
给定一个整数数组 A,找到 min(B) 的总和,其中 B 的范围为 A 的每个(连续)子数组。
|
||||
|
||||
由于答案可能很大,因此返回答案模 10^9 + 7。
|
||||
|
||||
|
||||
## 解题思路
|
||||
|
||||
- 首先想到的是暴力解法,用两层循环,分别枚举每个连续的子区间,区间内用一个元素记录区间内最小值。每当区间起点发生变化的时候,最终结果都加上上次遍历区间找出的最小值。当整个数组都扫完一遍以后,最终结果模上 10^9+7。
|
||||
- 上面暴力解法时间复杂度特别大,因为某个区间的最小值可能是很多区间的最小值,但是我们暴力枚举所有区间,导致要遍历的区间特别多。优化点就在如何减少遍历的区间。第二种思路是用 2 个单调栈。想得到思路是 `res = sum(A[i] * f(i))`,其中 f(i) 是子区间的数,A[i] 是这个子区间内的最小值。为了得到 f(i) 我们需要找到 left[i] 和 right[i],left[i] 是 A[i] 左边严格大于 A[i](>关系)的区间长度。right[i] 是 A[i] 右边非严格大于(>=关系)的区间长度。left[i] + 1 等于以 A[i] 结尾的子数组数目,A[i] 是唯一的最小值;right[i] + 1 等于以 A[i] 开始的子数组数目,A[i] 是第一个最小值。于是有 `f(i) = (left[i] + 1) * (right[i] + 1)`。例如对于 [3,1,4,2,5,3,3,1] 中的“2”,我们找到的串就为[4,2,5,3,3],2 左边有 1 个数比 2 大且相邻,2 右边有 3 个数比 2 大且相邻,所以 2 作为最小值的串有 2 * 4 = 8 种。用排列组合的思维也能分析出来,2 的左边可以拿 0,1,…… m 个,总共 (m + 1) 种,同理右边可以拿 0,1,…… n 个,总共 (n + 1) 种,所以总共 (m + 1)(n + 1)种。只要计算出了 f(i),这个题目就好办了。以 [3,1,2,4] 为例,left[i] + 1 = [1,2,1,1],right[i] + 1 = [1,3,2,1],对应 i 位的乘积是 f[i] = [1 * 1,2 * 3,1 * 2,1 * 1] = [1,6,2,1],最终要求的最小值的总和 res = 3 * 1 + 1 * 6 + 2 * 2 + 4 * 1 = 17。
|
||||
- **看到这种 mod1e9+7 的题目,首先要想到的就是dp**。最终的优化解即是利用 DP + 单调栈。单调栈维护数组中的值逐渐递增的对应下标序列。定义 `dp[i + 1]` 代表以 A[i] 结尾的子区间内最小值的总和。状态转移方程是 `dp[i + 1] = dp[prev + 1] + (i - prev) * A[i]`,其中 prev 是比 A[i] 小的前一个数,由于我们维护了一个单调栈,所以 prev 就是栈顶元素。(i - prev) * A[i] 代表在还没有出现 prev 之前,这些区间内都是 A[i] 最小,那么这些区间有 i - prev 个,所以最小值总和应该是 (i - prev) * A[i]。再加上 dp[prev + 1] 就是 dp[i + 1] 的最小值总和了。以 [3, 1, 2, 4, 3] 为例,当 i = 4, 所有以 A[4] 为结尾的子区间有:
|
||||
|
||||
[3]
|
||||
[4, 3]
|
||||
[2, 4, 3]
|
||||
[1, 2, 4, 3]
|
||||
[3, 1, 2, 4, 3]
|
||||
在这种情况下, stack.peek() = 2, A[2] = 2。前两个子区间 [3] and [4, 3], 最小值的总和 = (i - stack.peek()) * A[i] = 6。后 3 个子区间是 [2, 4, 3], [1, 2, 4, 3] 和 [3, 1, 2, 4, 3], 它们都包含 2,2 是比 3 小的前一个数,所以 dp[i + 1] = dp[stack.peek() + 1] = dp[2 + 1] = dp[3] = dp[2 + 1]。即需要求 i = 2 的时候 dp[i + 1] 的值。继续递推,比 2 小的前一个值是 1,A[1] = 1。dp[3] = dp[1 + 1] + (2 - 1) * A[2]= dp[2] + 2。dp[2] = dp[1 + 1],当 i = 1 的时候,prev = -1,即没有人比 A[1] 更小了,所以 dp[2] = dp[1 + 1] = dp[-1 + 1] + (1 - (-1)) * A[1] = 0 + 2 * 1 = 2。迭代回去,dp[3] = dp[2] + 2 = 2 + 2 = 4。dp[stack.peek() + 1] = dp[2 + 1] = dp[3] = 4。所以 dp[i + 1] = 4 + 6 = 10。
|
||||
- 与这一题相似的解题思路的题目有第 828 题,第 891 题。
|
Reference in New Issue
Block a user