LeetCode 42

给定 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开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!**

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值