力扣42 接雨水 (力扣11 盛最多水的容器 扩展)

力扣42 接雨水 (力扣11 盛最多水的容器 扩展)

lucas Lv4

题目

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。

示例 1

1.png

输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]
输出:6
解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。

示例 2

输入:height = [4,2,0,3,2,5]
输出:9

提示

n == height.length
1 <= n <= 2 * 104
0 <= height[i] <= 105

分析

上一题 相比,这一题要求考虑柱子整体的排列规律

而对于单个柱子能放下多少水,需要考虑左边最大柱子和右边最大柱子 中较小的那个,减去当前柱子的高度,也就是这样
2.png
图可见,绿色柱子是由两个红柱子挡起来的,并且绿柱子位置是两个红柱子较小的那个高度决定的

由此我们就可以编写代码
记录下每个柱子的左边最高柱子和右边最高柱子,再遍历数组进行减法即可

1
2
3
4
// i 位置的左边最高柱子,要么左临元素的左边最高柱子,要么是他自己
// i 位置的右边最高柱子,要么右临元素的左边最高柱子,要么是他自己
lMax[i] = maxOf(lMax[i - 1], height[i])
rMax[i] = maxOf(rMax[i + 1], height[i])

为什么要考虑他自己呢?
因为最高的柱子如果不考虑自己,minOf(lMax,rMax) - height 就会是负值
所以我们编码如下

代码

第一种

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class Solution {
fun trap(height: IntArray): Int {
val lMax = IntArray(height.size)
val rMax = IntArray(height.size)
lMax[0] = height[0]
rMax[height.lastIndex] = height[height.lastIndex]
var left:Int
var right:Int
for (i in 1..height.lastIndex) {
left = i
right = height.lastIndex - i
lMax[left] = maxOf(lMax[left - 1], height[left])
rMax[right] = maxOf(rMax[right + 1], height[right])
}
var count = 0
for (i in height.indices) {
count += (minOf(rMax[i], lMax[i]) - height[i])
println(count)
}
}
}

这个时间复杂度是 O(n) ,空间复杂度是 O(n)

优化后

其实我们不需要确定两个最大值,只需确定一个最大值,并且另一个最大值一定比这个大,就可以了
举个例子,我们知道一个柱子 lMax 是 2 ,而不知道 rMax,但是知道 rMax 肯定 >=2 ,所以 minOf(lMax,rMax) 的结果就已经可以确定了

这样我们就可以将记录 lMax rMax 的数组空间节省出来,并且立刻算出结果,不用最后再遍历一遍了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class Solution {
fun trap(height: IntArray): Int {
if (height.size <= 2) {
return 0
}
var lM = 0
var rM = 0
var left = 0
var right = height.lastIndex
var count = 0
while (left < right) {
lM = maxOf(lM, height[left])
rM = maxOf(rM, height[right])
if (lM < rM) {
count += lM - height[left]
left++
} else {
count += rM - height[right]
right--
}
}
return count
}
}

这个时间复杂度是 O(n) ,空间复杂度是 O(1)

总结

没有总结

  • Title: 力扣42 接雨水 (力扣11 盛最多水的容器 扩展)
  • Author: lucas
  • Created at : 2023-04-18 21:41:43
  • Updated at : 2024-11-11 16:56:54
  • Link: https://darkflamemasterdev.github.io/2023/04/18/力扣42:接雨水(力扣11-盛水最多的容器-扩展)/
  • License: This work is licensed under CC BY-NC-SA 4.0.
Comments
On this page
力扣42 接雨水 (力扣11 盛最多水的容器 扩展)