给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。
输入: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 个单位的雨水(蓝色部分表示雨水)。
输入:height = [4,2,0,3,2,5]
输出:9
====================================================================
枚举每一个"凹坑",然后向后向右找最高的柱子二者间的最小值。
二者间的较小值减去这个“坑”的高度就是存储的水。
时间复杂度 O ( n 2 ) O(n^2) O(n2)
注意,这里的“水”的累加,是按竖着的方块计算的。
如下图:
class Solution {
public:
int trap(vector& height) {
int n = height.size();
if(n<=2) return 0;
int ans = 0;
for(int i=1;i<n-1;i++) {
int h1 = 0, h2 = 0;
for(int j=i;j>=0;j–) h1 = max(h1,height[j]);
for(int j=i;j<n;j++) h2 = max(h2,height[j]);
ans += min(h1,h2) - height[i];
}
return ans;
}
};
类似于前缀和的思想,预处理一下。
时间复杂度: O ( n ) O(n) O(n)
class Solution {
public:
int trap(vector& height) {
int ans = 0,n = height.size();
if(n<=2) return 0;
vector lm(n),rm(n);
for(int i=0;i<n;i++){
if(i == 0) lm[i] = height[i];
else lm[i] = max(lm[i-1],height[i]);
}
for(int i=n-1;i>=0;i–){
if(i == n-1) rm[i] = height[i];
else rm[i] = max(rm[i+1],height[i]);
}
for(int i=1;i<n-1;i++){
ans += min(lm[i],rm[i])-height[i];
}
return ans;
}
};
上面是从左向右遍历来累加水柱。
如果考虑从左右两边来遍历的话,就可以这样解决。
记 leftmax, rightmax 分别为从左到右遍历、从右到左遍历的最大值,
而left、right 分别是从左到右、从右到左的指针,
这个时候 ,left的水柱对答案对贡献的为 m i n ( l e f t m a x , r i g h t m a x ) min(leftmax,rightmax) min(leftmax,rightmax);
但是此时leftmax的值已经求出,但是rightmax的值尚未求出,所以,如果此时leftmax小于等于rightmax的话,left的水柱对答案的贡献就知道了,然后更新leftmax,left右移。
反之亦然。
class Solution {
public:
int trap(vector& height) {
int ans = 0, n = height.size();
if(n<=2) return 0;
int l = 1 , r = n - 2;
int leftmax = height[0], rightmax = height[n-1];
while(l <= r){
if(leftmax <= rightmax){
ans += max(0,leftmax - height[l]);
leftmax = max(leftmax,height[l++]);
}else{
ans += max(0,rightmax - height[r]);
rightmax = max(rightmax,height[r–]);
}
}
return ans;
}
};
====================================================================
换一种分割思路,将“水”按行进行划分。如下图:
时间复杂度: O ( m a x H ∗ n ) O(maxH*n) O(maxH∗n)
class Solution {
public:
int trap(vector& height) {
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
1200页Java架构面试专题及答案
小编整理不易,对这份1200页Java架构面试专题及答案感兴趣劳烦帮忙转发/点赞
百度、字节、美团等大厂常见面试题
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
[外链图片转存中…(img-sb0Zvfop-1713451872680)]
[外链图片转存中…(img-trlYnLpL-1713451872682)]
百度、字节、美团等大厂常见面试题
[外链图片转存中…(img-2O10Gk0z-1713451872684)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!