leetcode题解-接雨水问题

2024-08-27 04:58
文章标签 leetcode 问题 题解 雨水

本文主要是介绍leetcode题解-接雨水问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 问题描述
  • 问题抽象
  • 问题关键点分析
  • 思路一
  • 思路二
  • 结束语

雨纷纷,旧故里草木深。
我听闻,你始终一个人。

今天忘忧跟大家一起探索一个跟雨水有关的算法,题目来源leetcode。

问题描述

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。下面是由数组 [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

问题抽象

这个问题可以抽象成一个网格,共有m(最高的柱子) * n(柱子数量)个格子,求哪些格子可以装水。如下图所示:

问题关键点分析

  • 本题的关键点在于,具备什么条件的格子可以存水?
  • 图中3 * 12个格子共被分为了三类。黑色部分已被柱子所占据,故不能存水。白色部分的左侧或者右侧,至少有一侧没被柱子挡住,最终水会流失掉。蓝色部分,未被柱子占据且左右侧均有柱子可以挡住水的流失,故能存水。由此得出结论,能存水的位置必须具备两个条件:(1) 空间未被柱子占据;(2) 左右两侧均有比当前位置高的柱子。

思路一

  1. 求出最高的柱子和最低的柱子;
  2. 从最低的柱子的高度开始遍历,遍历当前高低下所有的位置(方格),判断当前方格是否没有柱子且左右均有大于等于当前位置高度的柱子。
  3. 层级加1,循环2的操作,直至层级大于最高柱子的层级结束。

代码实现:

public static int trap(int[] height) {//1. 求出最高的和最低的柱子if(height.length == 0){return 0;}int min = height[0];int max = height[0];for (int i : height) {if(i > max) {max = i;}if(i < min){min = i;}}//记述,表示总计可以接住的雨水数量int sum = 0;//从最底层开始遍历,遍历到大于最高层结束for (int i = min; i <= max; i++){//遍历当前层级的所有位置(方格)//省略掉第一个位置和最后一个位置的判断,因为第一个位置和最后一个位置不会存水for(int j = 1; j < height.length - 1; j++){//判断当前位置是否被柱子占据if(height[j] < i){//标示左侧是否有大于等于当前位置的柱子boolean left = false;boolean right = false;for (int k = 0; k < height.length; k++){//判断左侧是否有柱子大于等于当前位置if(k < j && height[k] >= i){left = true;}//判断右侧是否有柱子大于等于当前位置if(k > j && height[k] >= i){right = true;}}//如果左右均有柱子遮挡,总数量加一if(left && right){sum ++;}}}}return sum;
}

运行结果:

ps:搞定!but,提交leetcode后,发现代码欠佳呀~

死在了最后一个测试用例上。

思路二

  1. 找出最高的柱子的高度和位置。
  2. 对最高柱子左侧的所有空格做判断,此时只需要判断左侧有没有比当前位置高的即可,右侧同理;此时的时间复杂度仅仅为O(n),远小于思路一的O(n^2);

代码实现:

public static int trap(int[] height){if(height.length == 0){return 0;}//1. 遍历一遍,找出最高柱子的坐标int maxIndex = 0;for (int i = 0; i < height.length; i++) {if(height[i] > height[maxIndex]){maxIndex = i;}}int sum= 0;//遍历最高柱左侧if(maxIndex > 1){int leftMax = 0;for (int i = 1; i < maxIndex; i++){//当前柱子高度小于左侧最高柱if (height[i] < height[leftMax]) {//该位置可以存放height[leftMax] - height[i]的水sum += height[leftMax] - height[i];}else{//否则标记当前坐标为左侧最高柱leftMax = i;}}}//遍历最高柱左侧if(maxIndex < height.length - 2){int rightMax = height.length - 1;//从最右侧开始,每次往左移动一个位置,遍历到最高柱for (int i = height.length - 2; i > maxIndex; i--){if (height[i] < height[rightMax]) {sum += height[rightMax] - height[i];}else{rightMax = i;}}}return sum;
}

运行结果:

提交结果:

搞定!当然,这个思路不一定是最优解,这个算法本身也存在这一些可以优化的地方,如果大家感兴趣,可以深入研究,在此忘忧就不再继续深入了。

PS:忘忧很期待与大家的交流,大家有任何问题可以通过公众号给忘忧留言,忘忧都会认真回复的,期望与大家在交流中一起成长!

结束语

忘忧的个人公众号,欢迎大家一起交流:算法之灵魂拷问

长风破浪会有时,直挂云帆济沧海!

这篇关于leetcode题解-接雨水问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1110703

相关文章

mybatis和mybatis-plus设置值为null不起作用问题及解决

《mybatis和mybatis-plus设置值为null不起作用问题及解决》Mybatis-Plus的FieldStrategy主要用于控制新增、更新和查询时对空值的处理策略,通过配置不同的策略类型... 目录MyBATis-plusFieldStrategy作用FieldStrategy类型每种策略的作

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

Python Jupyter Notebook导包报错问题及解决

《PythonJupyterNotebook导包报错问题及解决》在conda环境中安装包后,JupyterNotebook导入时出现ImportError,可能是由于包版本不对应或版本太高,解决方... 目录问题解决方法重新安装Jupyter NoteBook 更改Kernel总结问题在conda上安装了

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

解决jupyterLab打开后出现Config option `template_path`not recognized by `ExporterCollapsibleHeadings`问题

《解决jupyterLab打开后出现Configoption`template_path`notrecognizedby`ExporterCollapsibleHeadings`问题》在Ju... 目录jupyterLab打开后出现“templandroidate_path”相关问题这是 tensorflo

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

Java多线程父线程向子线程传值问题及解决

《Java多线程父线程向子线程传值问题及解决》文章总结了5种解决父子之间数据传递困扰的解决方案,包括ThreadLocal+TaskDecorator、UserUtils、CustomTaskDeco... 目录1 背景2 ThreadLocal+TaskDecorator3 RequestContextH

关于Spring @Bean 相同加载顺序不同结果不同的问题记录

《关于Spring@Bean相同加载顺序不同结果不同的问题记录》本文主要探讨了在Spring5.1.3.RELEASE版本下,当有两个全注解类定义相同类型的Bean时,由于加载顺序不同,最终生成的... 目录问题说明测试输出1测试输出2@Bean注解的BeanDefiChina编程nition加入时机总结问题说明

关于最长递增子序列问题概述

《关于最长递增子序列问题概述》本文详细介绍了最长递增子序列问题的定义及两种优化解法:贪心+二分查找和动态规划+状态压缩,贪心+二分查找时间复杂度为O(nlogn),通过维护一个有序的“尾巴”数组来高效... 一、最长递增子序列问题概述1. 问题定义给定一个整数序列,例如 nums = [10, 9, 2