当前位置: 首页 > news >正文

代码随想录:动态规划4-5

42.接雨水

题目

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

示例 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 个单位的雨水(蓝色部分表示雨水)。 

示例 2:

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

代码

class Solution {public int trap(int[] height) {//雨水的关键是找凹槽,凹槽是栈顶的mid,右边柱子是当前更大的元素,左边的柱子是栈顶下面的元素//单调递减栈:寻找右边第一个更大的元素Stack<Integer> stack = new Stack<>();//栈中放的是元素下标,第一个元素入栈,高度=height[0]stack.push(0);int sum = 0;  //初始雨水//i是当前元素下标,height[i]是当前元素高度for(int i=1; i < height.length; i++){//当前元素高度 < 栈顶元素高度if(height[i] < height[stack.peek()]){stack.push(i);  //当前元素下标入栈}//当前元素高度 = 栈顶元素高度else if(height[i] == height[stack.peek()]){//因为栈顶高度和当前元素高度,栈顶的柱子肯定没有雨水stack.pop();  //栈顶元素入栈stack.push(i);  //当前元素下标入栈}//当前元素高度 > 栈顶元素高度,出现凹槽else{//只要当前元素高度比栈顶元素高度大,循环处理所有凹槽while(!stack.isEmpty() && height[i] > height[stack.peek()]){int mid = stack.pop();  //凹槽下标//mid出栈后,如果栈为空,说明mid左边是空的,没有雨水,[1],mid=1,1出栈就行//mid出栈后,如果栈非空,说明mid左边不空,有雨水,需要计算雨水量if(!stack.isEmpty()){int left = stack.peek();  //凹槽左边下标//mid凹槽的雨水高度 = 左右高度最小值-凹槽高度int h = Math.min(height[i], height[left]) - height[mid];//mid凹槽的雨水宽度int w = i - left - 1;  //这里要写- left - 1,不能写-mid,[4,2,0,3,2,5]画个图就知道了int area = h * w;  //mid凹槽雨水量sum += area;   }}stack.push(i);//当前元素下标入栈}}return sum;  //返回总雨水量}
}

84.柱状图中最大的矩形

题目

给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。

求在该柱状图中,能够勾勒出来的矩形的最大面积。

示例 1:

输入:heights = [2,1,5,6,2,3]
输出:10
解释:最大的矩形为图中红色区域,面积为 10

示例 2:

输入: heights = [2,4]
输出: 4

代码

class Solution {public int largestRectangleArea(int[] heights) {//关键是找变小的元素,变小了,就把前面的矩形面积计算出来//扩容,头尾加元素0int[] newHeights = new int[heights.length + 2];//头部加0,防止出现[8,6,4,2]递减序列算出来是0newHeights[0] = 0;//尾部加0,防止出现[2,4,6,8]递增序列算出来是0newHeights[newHeights.length - 1] = 0;//数组copyfor(int i=0; i < heights.length; i++){newHeights[i+1] = heights[i];}int res = 0; //初始最大面积//单调栈,找右边第一个更小的元素Stack<Integer> stack = new Stack<>();//第一个元素下标入栈stack.push(0);  for(int i=1; i < newHeights.length; i++){//当前元素高度 > 栈顶元素高度//举例:[1,5,6]持续入栈if(newHeights[i] > newHeights[stack.peek()]){stack.push(i);  //下标入栈}//当前元素高度 = 栈顶元素高度//举例:[1,5,5,6],当前元素=2,最大值会在第一个5取到,5不用出栈else if(newHeights[i] == newHeights[stack.peek()]){stack.push(i);  //下标入栈}//当前元素高度 < 栈顶元素高度else{//[1,5,6],当前元素=2,要出栈循环计算矩形大小while(!stack.isEmpty() && newHeights[i] < newHeights[stack.peek()]){int mid = stack.pop();  //6的下标if(!stack.isEmpty()){int left = stack.peek();  //5的下标int h = newHeights[mid]; //高度是6int w = i - left - 1;  //宽度是1int area = h * w; //面积是6res = Math.max(res, area);  //更新最大矩形面积}}stack.push(i);}}return res;}
}

http://www.mrgr.cn/news/28928.html

相关文章:

  • Java Stream流操作List全攻略:Filter、Sort、GroupBy、Average、Sum实践
  • MC1.12.2 macOS高清修复OptiFine运行崩溃
  • RabbitMQ介绍与使用
  • 服务器一次性部署One API + ChatGPT-Next-Web
  • 2025华数杯国际赛A题完整论文讲解(含每一问python代码+数据+可视化图)
  • 数据库事务
  • 安徽省建设工程企业资质管理新动向
  • 阿里OSS对象存储服务,实现图片上传回显
  • 2019-2023(CSP-J)选择题真题解析
  • 【Android】使用Room数据库解决本地持久化
  • 每日一题——第九十八题
  • 【 html+css 绚丽Loading 】000051 方寸轮回矩
  • java138-异常处理_java 138错误
  • 2022高教社杯全国大学生数学建模竞赛C题 问题二(1) Python代码
  • 2022高教社杯全国大学生数学建模竞赛C题 问题一(1) Python代码
  • linux 操作系统下的 declare 命令介绍和使用案例
  • 【零散技术】详解Odoo17邮件发送(一)
  • 【乐企】基础版本开票代码接口声明
  • 【Git】将本地项目上传到git | 在IDEA的提交记录中更改 提交的用户名
  • vue MVC设计模式与MVVM设计模式
  • 牛客周赛 Round 60(下)
  • C++:初始化列表
  • nginx服务器安装和部署代理
  • Weapons Armor PBR Pack 1 - Fantasy RPG 武器护甲游戏模型
  • 【3D打印】常用Gcode和相关示例
  • a,b,c中的最大值