代码随想录算法训练营第五十九天| 503.下一个更大元素II 42. 接雨水

文档讲解:代码随想录

视频讲解:代码随想录B站账号

状态:看了视频题解和文章解析后做出来了

503.下一个更大元素II

class Solution:
    def nextGreaterElements(self, nums: List[int]) -> List[int]:
        res = [-1] * len(nums)
        stack = []

        for i in range(len(nums)*2):
            while len(stack) > 0 and nums[i%len(nums)] > nums[stack[-1]]:
                res[stack[-1]] = nums[i%len(nums)]
                stack.pop()
            stack.append(i%len(nums))
        return res
  • 时间复杂度:O(n)
  • 空间复杂度:O(n)

这道题和每日温度非常相似,只不过这次的数组是头接尾的,也就是说后面的元素也可以视前面的元素为比自己大的元素。

第一种方法是复制一个nums数组然后和原数组拼接在一起,最后把res resize成原来的大小,虽然最后的resize复杂度为O(1),但扩充数组的复杂度就是O(n)了,不太值得。

比较好的一种方式是循环这个数组两遍,所以在for循环的range里,使用的是len(nums) * 2。

在提取当前元素的值时,需要用到取模操作,取nums的长度,也就是说假设nums长度为5,那么循环到6的时候,下标为1。这个取模操作要应用到所有和 i 相关的操作中防止逾越index问题。

42. 接雨水 

暴力双指针

class Solution:
    def trap(self, height: List[int]) -> int:
        res = 0

        for i in range(1, len(height)-1):
            left_bar, right_bar = height[i], height[i]

            for j in range(i+1, len(height)):
                if height[j] > right_bar:
                    right_bar = height[j]
            for z in range(i-1, -1, -1):
                if height[z] > left_bar:
                    left_bar = height[z]
            
            h = min(right_bar, left_bar) - height[i]
            res += h

        return res
  • 时间复杂度:O(n^2)
  • 空间复杂度:O(n)

可以从行和列接雨水,从列接雨水的逻辑比较清晰,因为宽度一定是1所以只需要计算高度就行了。

想要计算一列能够接多少雨水,和木桶问题很像,取决于两边较矮的那个边的长度。

所以对于每一列,只需要首先将left_bar,right_bar初始化为当前的高度,然后依次向左和向右遍历,只要找到比左bar和右bar更高的bar就更新。

最后计算的时候,取两者的较小然后减去当前的高度 h = min(right_bar, left_bar) - height[i]。

卡哥的代码里最后还加了个判断h大于0的if,这个没必要哈因为left_bar和right_bar只有比height[i]大的时候才更新,没找到就是它自己,所以最差情况也是0,不可能小于0。

双指针优化

class Solution:
    def trap(self, height):
        if len(height) <= 2:
            return 0

        size = len(height)
        max_left = [0] * size
        max_right = [0] * size

        # Record the maximum height to the left of each bar
        max_left[0] = height[0]
        for i in range(1, size):
            max_left[i] = max(height[i], max_left[i - 1])

        # Record the maximum height to the right of each bar
        max_right[size - 1] = height[size - 1]
        for i in range(size - 2, -1, -1):
            max_right[i] = max(height[i], max_right[i + 1])

        # Calculate the total trapped water
        total_water = 0
        for i in range(size):
            water_at_bar = min(max_left[i], max_right[i]) - height[i]
            total_water += water_at_bar

        return total_water
  • 时间复杂度:O(n)
  • 空间复杂度:O(n)

在暴力方法中,我们每到一个位置都要计算它左边和右边的最高bar,这里就会涉及很多重复计算。如果我们把每个点的左右最高bar首先计算出来保存在两个数组中,在循环过程中使用这个数组中的值就避免了重复运算。

单调栈

class Solution:
    def trap(self, height: List[int]) -> int:
        stack = [0]
        result = 0
        for i in range(1, len(height)):
            while stack and height[i] > height[stack[-1]]:
                mid_height = stack.pop()
                if stack:
                    # 雨水高度是 min(凹槽左侧高度, 凹槽右侧高度) - 凹槽底部高度
                    h = min(height[stack[-1]], height[i]) - height[mid_height]
                    # 雨水宽度是 凹槽右侧的下标 - 凹槽左侧的下标 - 1
                    w = i - stack[-1] - 1
                    # 累计总雨水体积
                    result += h * w
            stack.append(i)
        return result
  • 时间复杂度:O(n)
  • 空间复杂度:O(n)

首先明确单调栈是按行来接雨水的:

然后确定单调栈是递增还是递减,这道题中是从栈头到栈尾单调递增。因为如果遇到了比前一个高的元素,说明遇到了桶的右bar,这时候就要pop来进行操作了。

分三种情况:

1. 如果当前元素小于前一个元素,直接入栈并保持单调递增。

2. 如果当前元素等于前一个元素,5543的情况,我们在后续计算雨水体积的时候,肯定也是用右侧的两个5的第二个5来计算,所以先pop再append。

3. 如果当前元素大于前一个元素,开始操作了。

- 首先pop获取上一个元素的高度,赋值给mid变量

- 计算高度,高度为当前元素 i 和当前栈头,也就是 i - 2的高度的较小值

- 计算宽度,宽度就是当前元素下标 i 减去栈头,再加1。假设当前元素下标为5,栈头为3,那么

5 - 3 - 1 = 1,宽度也只是1,符合题意。

- 将长乘以宽的体积计入结果变量res中。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Vue性能优化方法

一、前言 1.1 为什么需要性能优化 用户体验&#xff1a;优化性能可以提升用户体验&#xff0c;降低加载时间和响应时间&#xff0c;让用户更快地看到页面内容。SEO优化&#xff1a;搜索引擎更喜欢快速响应的网站&#xff0c;优化性能可以提高网站的排名。节约成本&#xff1…

【知识】稀疏矩阵是否比密集矩阵更高效?

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.cn] 问题提出 有些地方说&#xff0c;稀疏图比密集图的计算效率更高&#xff0c;真的吗&#xff1f; 原因猜想 这里的效率高&#xff0c;应该是有前提的&#xff1a;当使用稀疏矩阵的存储格式(如CSR)时&#xff0c;计…

云时空社会化商业 ERP 系统 Shiro 反序列化漏洞复现

0x01 产品简介 时空云社会化商业ERP&#xff08;简称时空云ERP&#xff09; &#xff0c;该产品采用JAVA语言和Oracle数据库&#xff0c; 融合用友软件的先进管理理念&#xff0c;汇集各医药企业特色管理需求&#xff0c;通过规范各个流通环节从而提高企业竞争力、降低人员成本…

Linux相关--笔试和面试高频

Linux RedHat公司已经宣布停止维护CentOS服务器操作系统&#xff0c;可以选择华为开源的欧拉系统、阿里开源的龙蜥系统和腾讯开源的TencentOS系统 面试 几个基本的Linux命令 pwd #查看当前绝对路径 结果/home/stu touch / vi编辑器 #创建文件 mkdir -p /home/stu/test #当…

ESP32-Web-Server 实战编程- 使用 AJAX 自动更新网页内容

ESP32-Web-Server 实战编程- 使用 AJAX 自动更新网页内容 概述 什么是 AJAX &#xff1f; AJAX Asynchronous JavaScript and XML&#xff08;异步的 JavaScript 和 XML&#xff09;。 AJAX 是一种用于创建快速动态网页的技术。 传统的网页&#xff08;不使用 AJAX&#…

鸿蒙原生应用/元服务开发-AGC分发如何下载管理Profile

一、收到通知 尊敬的开发者&#xff1a; 您好&#xff0c;为支撑鸿蒙生态发展&#xff0c;HUAWEI AppGallery Connect已于X月XX日完成存量HarmonyOS应用/元服务的Profile文件更新&#xff0c;更新后Profile文件中已扩展App ID信息&#xff1b;后续上架流程会检测API9以上Harm…

直接套用的软件详细设计说明书

软件开发全套资料过去进主页&#xff01;

stm32 计数模式

计数模式 但是对于通用定时器而言&#xff0c;计数器的计数模式不止向上计数这一种。上文基本定时器中计数器的计数模式都是向上计数的模式。 向上计数模式&#xff1a;计数器从0开始&#xff0c;向上自增&#xff0c;计到和自动重装寄存器的目标值相等时&#xff0c;计数器清…

安卓apk抓包

起因 手机&#xff08;模拟器&#xff09;有时候抓不到apk的包&#xff0c;需要借助Postern设置一个代理&#xff0c;把模拟器的流量代理到物理机的burp上。 解决方案 使用Postern代理&#xff0c;把apk的流量代理到burp。 Postern是一个用于代理和网络流量路由的工具&#xf…

Apache Flink(三):Flink核心特性及应用场景

&#x1f3e1; 个人主页&#xff1a;IT贫道_大数据OLAP体系技术栈,Apache Doris,Clickhouse 技术-CSDN博客 &#x1f6a9; 私聊博主&#xff1a;加入大数据技术讨论群聊&#xff0c;获取更多大数据资料。 &#x1f514; 博主个人B栈地址&#xff1a;豹哥教你大数据的个人空间-豹…

Linux服务器SSH客户端断开后保持程序继续运行的方法

目录 1. nohup 命令&#xff1a; 2. tmux 或 screen&#xff1a; 3 final shell 断开后服务器如何继续执行令&#xff1f; 方法一&#xff1a;使用 nohup 命令 方法二&#xff1a;将命令放在后台执行 4 你可以使用 jobs 命令查看当前终端中正在后台运行的任务 &#xff…

【Linux | Docker】内网穿透实现远程访问Nginx Proxy Manager

文章目录 前言1. docker 一键安装2. 本地访问3. Linux 安装cpolar4. 配置公网访问地址5. 公网远程访问6. 固定公网地址 前言 Nginx Proxy Manager 是一个开源的反向代理工具&#xff0c;不需要了解太多 Nginx 或 Letsencrypt 的相关知识&#xff0c;即可快速将你的服务暴露到外…

C++设计模式——原型 (克隆)模式

一、什么是原型模式 Prototype模式说简单点&#xff0c;就是提供了一个clone, 通过已存在对象进行新对象创建。clone&#xff08;&#xff09;实现和具体的实现语言相关&#xff0c;在C中我们通过拷贝构造函数实现。 那为啥要写clone的接口来实现这个目的呢&#xff1f;直接使…

关于2024年天津铁道职业技术学院专升本考试报名工作的通知

天津铁道职业技术学院关于2024年高职升本科报名工作的通知 根据市高招办关于2024年天津市高职升本科的工作安排&#xff0c;为做好天津铁道职业技术学院2024届毕业生高职升本科考试报名工作&#xff0c;现将相关事项通知如下&#xff1a; 1. 报考资格&#xff1a;2024届天津铁…

【EI会议征稿】第四届生物信息学与智能计算国际学术研讨会(BIC 2024)

第四届生物信息学与智能计算国际学术研讨会&#xff08;BIC 2024&#xff09; 2024 4th International Conference on Bioinformatics and Intelligent Computing 2024年第四届生物信息学与智能计算国际学术研讨会 &#xff08;BIC 2024&#xff09;将定于2024年1月26-28日在…

ora.LISTENER.lsnr状态为Not All Endpoints Registered

客户的监控反馈有个监听无法连接&#xff0c;登录环境检查发现ora.LISTENER.lsnr的状态为Not All Endpoints Registered&#xff0c;如下 [rootdb2 ~]# crsctl status res -t -------------------------------------------------------------------------------- NAME …

IDEA如何配置Git 遇到问题的解决

新建项目 点击 会变红 会生成.git隐藏文件 配置远程仓库路径&#xff1a;点击Manage Remotes&#xff1a;将远程仓库的链接放到这里&#xff1a; 得到如下样式&#xff1a; 此时提交到本地仓库 点击add&#xff0c;添加到暂存文件&#xff1a; 此时文件变绿&#xf…

5 面试题--redis

伪客户端&#xff1a; 伪客户端的 fd 属性值为 -1&#xff1b;伪客户端处理的命令请求来源于 AOF ⽂件或者 Lua 脚本&#xff0c;⽽不是⽹络&#xff0c;所以这种客户端不需要套接字连接&#xff0c;⾃然也不需要记录套接字描述符。⽬前 Redis 服务器会在两个地⽅ ⽤到伪客户端…

【Web】NewStarCTF Week3 个人复现

①Include &#x1f350; ?filephpinfo 提示查下register_argc_argv 发现为on LFI包含 pearcmd命令执行学习 pearcmd.php文件包含妙用 ?file/usr/local/lib/php/pearcmd&config-create/<?eval($_POST[a])?>./ha.php ?file./ha post传&#xff1a; asystem…

云时空社会化商业 ERP 系统 service SQL 注入漏洞复现

0x01 产品简介 时空云社会化商业ERP&#xff08;简称时空云ERP&#xff09; &#xff0c;该产品采用JAVA语言和Oracle数据库&#xff0c; 融合用友软件的先进管理理念&#xff0c;汇集各医药企业特色管理需求&#xff0c;通过规范各个流通环节从而提高企业竞争力、降低人员成本…