【算法】接雨水

news2025/1/11 9:07:19

难度:困难

题目

给定 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

提示:

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

解题思路:

这道题目的核心是求解一个数组中,可以储存多少单位的雨水。想象一下,每个数组元素代表一块宽度为1的柱子高度,雨水会在柱子之间累积。我们要找到所有柱子间的凹陷部分可以储存的雨水总量。

  1. 初始化:设置两个指针left和right分别指向数组的开始和结束位置,同时初始化两个变量maxLeft和maxRight来跟踪左右两边的最大高度。
  2. 循环条件:当left < right时,进入循环。
  3. 更新最大高度:在每一步中,比较当前left和right位置的柱子高度与它们各自方向上的最大高度,更新maxLeft和maxRight。因为雨水的高度受限于两边的最低高度,所以我们只需要关注限制雨水高度的那一边。
  4. 计算并累加雨水:对于当前位置,雨水的高度是由Math.min(maxLeft, maxRight)决定的,减去当前柱子的高度,就是这部分可以储存的雨水量。累加这部分雨水到总雨水中。
  5. 移动指针:根据maxLeft和maxRight的大小关系,移动高度较小的那一侧的指针。因为那一侧的柱子高度限制了雨水的高度,移动它可以探索是否有更大的空间储存雨水。
  6. 重复步骤3-5,直到left >= right。
  7. 返回结果:最后返回累加的雨水总量。

JavaScript实现:

function trap(height) {
    let left = 0, right = height.length - 1;
    let maxLeft = 0, maxRight = 0;
    let totalRainwater = 0;

    while (left < right) {
        if (height[left] <= height[right]) {
            // 左侧柱子矮或者相等时,更新左侧最大高度并移动left指针
            maxLeft = Math.max(maxLeft, height[left]);
            totalRainwater += maxLeft - height[left];
            left++;
        } else {
            // 右侧柱子矮时,更新右侧最大高度并移动right指针
            maxRight = Math.max(maxRight, height[right]);
            totalRainwater += maxRight - height[right];
            right--;
        }
    }

    return totalRainwater;
}

// 示例
//console.log(trap([0,1,0,2,1,0,1,3,2,1,2,1])); // 输出: 6

使用双指针法解决“计算柱子间雨水量”问题的原因主要在于其高效性和直观性,具体优势如下:

  1. 效率高:双指针法允许我们在一次遍历中解决问题,只需两个指针分别从数组的两端开始向中间移动,时间复杂度为O(n),其中n为柱子的数量。相比于其他可能需要多次遍历或使用额外空间的算法,双指针法更加高效。
  2. 减少空间复杂度:此方法不需要额外的大空间来存储信息,除了几个变量外,几乎不需要额外的空间开销,因此空间复杂度为O(1),非常节省内存。
  3. 直观易懂:双指针策略直观地模拟了从数组两端向中心逐步逼近的过程,容易理解。通过比较左右两侧的最大高度并移动较矮一侧的指针,可以直观地看到如何逐步累加雨水量。
  4. 自动去重:在处理连续相同高度的柱子时,双指针法可以自然地跳过这些柱子,避免了重复计算。例如,当一个指针向内移动时,如果当前柱子高度小于或等于最大高度,直接计算差值并移动指针,不会重复考虑相同高度的柱子。
  5. 灵活性:双指针法可以适应多种变体问题,比如如果问题变为计算两边高度不同时的雨水量,只需稍作调整即可,体现了算法的灵活性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1921384.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

开放式耳机哪款好一点?开放式耳机科普五款推荐!

“选择开放式耳机真的太难了” “哥&#xff0c;怎么才能选到心仪的开放式耳机啊” 这种评论总是会出现后台或者现实的朋友也会问起来&#xff0c;所以作为耳机测评的博主&#xff0c;在这里给大家科普一下到底一款好用的开放式耳机到底怎么选&#xff0c;这篇文章我花了三天…

【RHCE】基于密钥身份认证

1.在本主机点击【⼯具】按钮打开【⽤⼾密钥管理者】选项 在该⻚⾯单击【⽣成】按钮来创建密钥对 在向导⻚⾯&#xff0c;选择【密钥类型】和【密钥⻓度】&#xff0c;这⾥保持默认。单击【下⼀步】按钮。 跳转到以下⻚⾯&#xff0c;表⽰密钥对已经创建完成。单击【下⼀步】给…

Nessus相关

tenable 1 安装nessus scanner 1 )安装nessus scanner: 方法一 curl -H X-Key: xxxxx https://cloud.tenable.com/install/scanner?namescanner-name&groupsscanner-group | bash方法二&#xff1a; **# for ubuntu, its https://www.tenable.com/downloads/api/v1/pu…

面试篇-Java-3+类加载+JVM 内存划分

文章目录 前言一、你知道类的加载过程吗1.1 你都知道哪些类型的类加载器1.2 你知道双亲委派机制吗1.3 你来说类加载都有哪些步骤1.3.1 加载&#xff1a;1.3.2 连接&#xff1a;1.3.1.2 验证1.3.1.3 准备1.3.1.4 解析 1.3.3 初始化 二、你知道JVM 中内存的分配吗2.1 你知道类加载…

通用图形处理器设计GPGPU基础与架构(二)

一、前言 本系列旨在介绍通用图形处理器设计GPGPU的基础与架构&#xff0c;因此在介绍GPGPU具体架构之前&#xff0c;需要了解GPGPU的编程模型&#xff0c;了解软件层面是怎么做到并行的&#xff0c;硬件层面又要怎么配合软件&#xff0c;乃至定出合适的架构来实现软硬件协同。…

如何修复d3dx9_43.dll文件丢失问题,实测有效的几种方法分享

在日常工作生活中&#xff0c;计算机可能会出现提示“d3dx9_43.dll文件丢失”的情况&#xff0c;面对这种情况&#xff0c;您该如何应对呢&#xff1f;这种情况在计算机使用过程中是比较常见的&#xff0c;对于经常使用计算机的人来说&#xff0c;难免会遇到这种情况。今天&…

Milvus 核心设计(1) ---- 数据一致性的等级及使用场景

目录 背景 Milvus的数据一致性 设置数据一致性等级 等级类型 PACELC定理 level 详细解释 Strong Bounded staleness Session Eventually 总结 背景 分布式上的可扩展性是个比较重要的concept。Chroma 核心之前写过了,他的最大优势在于轻量级且好用。Milvus相对Ch…

PHP企业工商年报大师微信小程序系统源码

&#x1f31f;轻松搞定年报难题&#xff01;&#x1f4bc; &#x1f680;【一键直达&#xff0c;年报不再繁琐】 还在为每年的企业工商年报而头疼吗&#xff1f;繁琐的表格、复杂的流程&#xff0c;让人望而却步&#xff1f;现在有了“企业工商年报大师”微信小程序&#xff…

彻底解决找不到msvcr120.dll,无法继续执行代码的问题(最新方法)

在使用电脑过程中经常会遇到各种问题&#xff0c;其中msvcr120.dll丢失或找不到msvcr120.dll问题就是常见之一&#xff0c;那么遇到msvcr120.dll丢失要怎么解决&#xff1f;msvcr120.dll又是什么为什么会丢失&#xff1f;今天给大家介绍一下msvcr120.dll文件跟msvcr120.dll丢失…

189. 轮转数组 --- 多种方式解题

直接开新数组&#xff0c;暴力遍历&#xff0c;新位置(原位置k)%nums.length class Solution {public void rotate(int[] nums, int k) {int[] ans new int[nums.length];for (int i 0; i < nums.length; i) {int pos (i k) % nums.length;ans[pos] nums[i];}for (int…

Ubuntu系统上安装Apache和WordPress

** 第一步跟新系统包 ** 首先跟新系统包 sudo apt update sudo apt upgrade第二步下载安装apache sudo apt install apache2 ##查看apache的状态是否启动成功 sudo systemctl status apache2 ##查看服务器的ip地址 sudo ip a通过ip地址进行访问apache页面 第三步下载安装…

常见条件控制算法流程图

内容讲解&#xff1a;流程控制[if…else…(if…elif…else…),while,for] 常见条件控制算法流程图高清图

k8s集群利用svc,ep代理另一个集群的svc服务,让集群正常调用

前提&#xff1a;两套集群的网络互通 背景&#xff1a;客户给两套集群&#xff0c;不通网络环境&#xff0c;但是两套集群中服务需要有调用故用此方法 A集群&#xff08;未部署服务a&#xff0c;但是部署了服务b,c,d&#xff09; B集群 &#xff08;只部署了服务a&#xff09; …

216.Mit6.S081-实验四-Traps

本实验探索如何使用陷阱实现系统调用。您将首先使用栈做一个热身练习&#xff0c;然后实现一个用户级陷阱处理的示例。 开始编码之前&#xff0c;请阅读xv6手册的第4章和相关源文件&#xff1a; kernel/trampoline.S&#xff1a;涉及从用户空间到内核空间再到内核空间的转换的…

检索 Postgres 不同版本功能差异的神器

官方 官方提供了 Feature Matrix https://www.postgresql.org/about/featurematrix pgPedia https://pgpedia.info/ 提供了更加丰富的内容 带搜索功能 不同版本下的 SQL 命令支持 系统表的支持 总结 横向对比其他数据库&#xff0c;官方的版本功能比对已经做的很不错了。pgP…

配置Redis时yml的格式导致报错

报错如下 java.lang.IllegalStateException: Failed to load ApplicationContext at org.springframework.test.context.cache.DefaultCacheAwareContextLoaderDelegate.loadContext(DefaultCacheAwareContextLoaderDelegate.java:98) at org.springframework.test.context.su…

这几个伦敦金交易常识性的知识 一定要知道

常识是指在伦敦金交易中&#xff0c;获得大部分投资者一致公认的有用的的一些投资知识&#xff0c;掌握这些知识&#xff0c;对我们做伦敦金交易是很重要的。下面我们就来讨论三个伦敦金交易的常识。 支撑阻力位。在伦敦金交易中支撑阻力位的重要性不言而喻。这里需要说明的是&…

YOLOv10: Real-Time End-to-End Object Detection

双重标签分配 与一对一多分配不同&#xff0c;一对一匹配只为每个地面真相分配一个预测&#xff0c;避免了NMS后处理。然而&#xff0c;这导致了较弱的监督&#xff0c;导致次优的准确性和收敛速度。幸运的是&#xff0c;这种缺陷可以通过一对一多分配来弥补。为此&#xff0c…

一起来学习孟德尔随机化临床医学SCI发表吧!!!

如今&#xff0c;临床科研工作者面对越来越重的科研压力&#xff0c;以及越来越高的 SCI 文章要求&#xff0c;如何才能在不 进实验室、不做基础科研的前提下&#xff0c;利用好各种公共数据资源快速发表 SCI 论著&#xff1f;这是一个困绕每一 个临床科研医生的话题。真正的随…

Docker 使用基础(4)—存储卷

&#x1f3ac;慕斯主页&#xff1a;修仙—别有洞天 ♈️今日夜电波&#xff1a;秒針を噛む—ずっと真夜中でいいのに。 0:34━━━━━━️&#x1f49f;──────── 4:20 &#x1f504; ◀️ ⏸ …