代码随想录算法训练营Day55|42.接雨水、84.柱状图中最大的矩形

news2025/1/23 10:45:42

接雨水

42. 接雨水 - 力扣(LeetCode)

暴力解法

对计算接到的雨水,有两种方式,一是按照行来计算。

另一种是按列计算

按列计算容易不乱。基本思路如下:

对每列i进行循环,在循环中,找到该列左侧的最大高度leftHeight和该列右侧的最大高度rightHeight,该列能接到的雨水为两者较小值减去该列的高度。

h = min(leftHeight,rightHeight) - height[i] 

当h大于0时,将结果累加进最后的返回结果result中.

具体思路可以看代码随想录 (programmercarl.com)

代码如下

class Solution {
public:
    //计算给定高度数组中能接的雨水量
    int trap(vector<int>& height) {
        int sum = 0; // sum 用于累计接到的雨水量
        for(int i = 0; i < height.size(); i++){ // 遍历数组中的每个元素
            if(i == 0 || i == height.size() - 1) // 数组的第一个和最后一个元素不能接雨水
                continue; // 跳过这两个元素
            int rightHeight = height[i]; // 右边最高的柱子高度,初始设为当前柱子高度
            int leftHeight = height[i]; // 左边最高的柱子高度,初始设为当前柱子高度
            for(int r = i + 1; r < height.size(); r++){ // 向右扫描,找到右边最高的柱子
                if(height[r] > rightHeight)
                    rightHeight = height[r];
            }
            for(int l = i - 1; l >= 0; l--){ // 向左扫描,找到左边最高的柱子
                if(height[l] > leftHeight)
                    leftHeight = height[l];
            }
            int h = min(leftHeight, rightHeight) - height[i]; // 计算当前位置可以接的雨水量
            if(h > 0)
                sum += h; // 如果可以接雨水,则加到总雨水量上
        }
        return sum; // 返回总的接雨水量
    }
};

算法的时间复杂度为O(n^2),对于每个元素都要向左右遍历一次获取最大的leftheight和rightheight,最后2个案例会超时,ac不了,空间复杂度为O(1)。

双指针法

在上述的暴力解法中,如果我们进行调试的话,会发现每次的运行都会左右遍历一次以获取最大的leftheight和rightheight,而这有很多的计算浪费,所以我们可以在开始就从左到右,从右到左遍历一次数组,得到两个数组,分别表示每列i的leftheight和rightheight,然后用上述计算h的方法再计算一次,将算法的时间复杂度降为O(n)。

class Solution {
public:
    int trap(vector<int>& height) {
        int sum = 0;
        vector<int> leftHeight(height.size(), 0); // 创建一个数组来存储每个位置的左边最高柱子高度
        vector<int> rightHeight(height.size(), 0); // 创建一个数组来存储每个位置的右边最高柱子高度
        int maxHeight_left_temp = height[0]; // 初始化左边最高柱子高度为第一个元素的高度
        int maxHeight_right_temp = height[height.size() - 1]; // 初始化右边最高柱子高度为最后一个元素的高度
        for (int i = 1; i < height.size(); i++) {
            maxHeight_left_temp = max(maxHeight_left_temp, height[i]); // 更新左边最高柱子高度
            leftHeight[i] = maxHeight_left_temp; // 存储左边最高柱子高度
        }
        for (int i = height.size() - 2; i >= 0; i--) {
            maxHeight_right_temp = max(maxHeight_right_temp, height[i]); // 更新右边最高柱子高度
            rightHeight[i] = maxHeight_right_temp; // 存储右边最高柱子高度
        }

        // 再次遍历数组,计算每个位置可以接的雨水量
        for (int i = 0; i < height.size(); i++) {
            int h = min(leftHeight[i], rightHeight[i]) - height[i]; // 计算当前位置可以接的雨水量
            if (h > 0)
                sum += h; // 如果可以接雨水,则加到总雨水量上
        }

        return sum;
    }
};

算法的时间复杂度为O(n),空间复杂度为O(n)(2个长度为n的数组),以空间换时间。

单调栈法

这里想像这样一个问题,接雨水即需要找到当前柱子右侧的大于等于该柱子高度的柱子,然后计算接到的雨水,因此,可以使用单调栈法。

从左到右遍历时,需要找到比当前元素大的元素,考虑使用单调递增栈(什么问题使用什么栈参考之前day52的文章)

class Solution {
public:
    int trap(vector<int>& height) {
        stack<int> st; // 创建一个栈来存储柱子的索引
        int sum = 0; // sum 用于累计接到的雨水量
        st.push(0); // 先将第一个柱子的索引入栈

        for (int i = 1; i < height.size(); i++) { // 从第二个柱子开始遍历数组
            if (height[i] < height[st.top()]) { // 如果当前柱子高度小于栈顶柱子高度
                st.push(i); // 将当前柱子的索引入栈
            } else if (height[i] == height[st.top()]) { // 如果当前柱子高度等于栈顶柱子高度
                st.pop(); // 弹出栈顶元素
                st.push(i); // 将当前柱子的索引入栈
            } else { // 如果当前柱子高度大于栈顶柱子高度
                while (!st.empty() && height[i] > height[st.top()]) { // 当栈不为空且当前柱子高度大于栈顶柱子高度时
                    int mid = st.top(); // 获取栈顶柱子的索引
                    st.pop(); // 弹出栈顶元素
                    if (!st.empty()) { // 如果栈不为空
                        int h = min(height[st.top()], height[i]) - height[mid]; // 计算凹槽的高度
                        int w = i - st.top() - 1; // 计算凹槽的宽度
                        sum += h * w; // 计算凹槽可以接的雨水量,并加到总雨水量上
                    }
                }
            }
            st.push(i); // 将当前柱子的索引入栈
        }

        return sum; // 返回总的接雨水量
    }
};

算法的时间复杂度为O(n),空间复杂度为O(n)。

柱状图中最大的矩形

84. 柱状图中最大的矩形 - 力扣(LeetCode)

暴力解法

类似上题的想法,对每个列i找到左方第一个高度小于它的列left和右方第一个高度小于它的列right,面积为(right - left - 1) *  heights[i]。同样,算法的时间复杂度为O(n^2),空间复杂度为O(1)。

class Solution {
public:
    int largestRectangleArea(vector<int>& heights) {
        int sum = 0; // sum 用于存储最大矩形面积
        for (int i = 0; i < heights.size(); i++) { // 遍历数组中的每个元素
            int left = i; // 初始化左边边界为当前柱子
            int right = i; // 初始化右边边界为当前柱子

            // 向左扫描,找到左边第一个高度小于当前柱子的位置
            for (left = i; left >= 0; left--) {
                if (heights[left] < heights[i]) break;
            }

            // 向右扫描,找到右边第一个高度小于当前柱子的位置
            for (right = i; right < heights.size(); right++) {
                if (heights[right] < heights[i]) break;
            }

            // 计算以当前柱子为高的矩形面积
            int w = right - left - 1; // 矩形的宽度
            int h = heights[i]; // 矩形的高,即当前柱子的高度
            sum = max(sum, w * h); // 更新最大矩形面积
        }
        return sum; // 返回最大矩形面积
    }
};

超时。

双指针法

有些难度,参考代码随想录代码随想录 (programmercarl.com)

class Solution {
public:
    int largestRectangleArea(vector<int>& heights) {
        vector<int> minLeftIndex(heights.size());
        vector<int> minRightIndex(heights.size());
        int size = heights.size();

        // 记录每个柱子 左边第一个小于该柱子的下标
        minLeftIndex[0] = -1; // 注意这里初始化,防止下面while死循环
        for (int i = 1; i < size; i++) {
            int t = i - 1;
            // 这里不是用if,而是不断向左寻找的过程
            while (t >= 0 && heights[t] >= heights[i]) t = minLeftIndex[t];
            minLeftIndex[i] = t;
        }
        // 记录每个柱子 右边第一个小于该柱子的下标
        minRightIndex[size - 1] = size; // 注意这里初始化,防止下面while死循环
        for (int i = size - 2; i >= 0; i--) {
            int t = i + 1;
            // 这里不是用if,而是不断向右寻找的过程
            while (t < size && heights[t] >= heights[i]) t = minRightIndex[t];
            minRightIndex[i] = t;
        }
        // 求和
        int result = 0;
        for (int i = 0; i < size; i++) {
            int sum = heights[i] * (minRightIndex[i] - minLeftIndex[i] - 1);
            result = max(sum, result);
        }
        return result;
    }
};

算法的时间复杂度为O(n),空间复杂度也为O(n)。

单调栈法

代码随想录 (programmercarl.com)

明天补

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

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

相关文章

python-逻辑语句

if else语句 不同于C&#xff1a;else if range语句&#xff1a; continue continue的作用是&#xff1a; 中断所在循环的当次执行&#xff0c;直接进入下一次 continue在嵌套循环中的应用 break 直接结束所在的循环 break在嵌套循环中的应用 continue和break&#xff0c;在…

【TB作品】atmega16 计算器,ATMEGA16单片机,Proteus仿真

实验报告&#xff1a;基于ATmega16单片机的简易计算器设计 1. 实验背景 计算器是日常生活和工作中不可或缺的工具&#xff0c;通过按键输入即可实现基本的四则运算。通过本实验&#xff0c;我们将利用ATmega16单片机、矩阵键盘和LCD1602显示屏&#xff0c;设计并实现一个简易…

【TB作品】智能台灯,ATMEGA16单片机,Proteus仿真

智能台灯 1 adc检测光强光敏电阻 显示电压 2 光强太高 也就是高于临界值 就关闭小灯 3 光强太低 也就是低于临界值 就打开小灯 3 按键修改临界值 显示 实验报告&#xff1a;基于ATMEGA16单片机的智能台灯设计与Proteus仿真 1. 实验背景 智能台灯是一种能够根据环境光强自动调…

【Altium】AD-焊盘介绍

【更多软件使用问题请点击亿道电子官方网站】 1、 文档目标 介绍PCB设计工具中焊盘的组成 2、 知识点 为元件创建封装时&#xff0c;焊盘都是不可获取的部分&#xff0c;一个完整的焊盘&#xff0c;包含了哪些部分&#xff0c;各自又是什么作用。 3、软硬件环境 1&#xff…

算法笔记:模拟过程(螺旋遍历矩阵)

1 模拟过程 “模拟过程题”通常指的是那些要求编程者通过编写代码来“模拟”或重现某个过程、系统或规则的题目。这类题目往往不涉及复杂的数据结构或高级算法&#xff0c;而是侧重于对给定规则的精确执行和逻辑的清晰表达。 其中螺旋遍历矩阵的题目就是一类典型的模拟过程题…

学习笔记(linux高级编程)7

2._exit 系统调用 void _exit(int status); 功能: 让进程退出,不刷新缓存区 参数: status:进程退出状态 返回值: 缺省 回调函数 3.atexit int atexit(void (*function)(void)); 功能: 注册进程退出前执行的函数 参数: function:函数指针 指向void返回值void参数的函数指针 返…

吴恩达《LangChain for LLM Application Development》课程笔记

目录 1. 前言 2. 课程笔记 2.1. 模型、提示和解析器 2.2. LLM记忆&#xff0c;上下文管理 2.3. 链式操作 2.4. 文档问答 2.4.1. stuff 方法 2.4.2. 其他方法 2.5. LLM应用评估 2.6. 代理 2.6.1. 预定义工具 2.6.2. 自定义工具 代码资源&#xff1a; 1. 前言 LangC…

汇聚荣拼多多电商好不好?

拼多多电商好不好?这是一个值得探讨的问题。拼多多作为中国领先的电商平台之一&#xff0c;以其独特的商业模式和创新的营销策略吸引了大量用户。然而&#xff0c;对于这个问题的回答并不是简单的好或不好&#xff0c;而是需要从多个方面进行综合分析。 一、商品质量 来看拼多…

混合专家模型(MoE)的前世今生

在文章《聊聊最近很火的混合专家模型&#xff08;MoE&#xff09;》中&#xff0c;我们简单介绍了MoE模型的定义和设计&#xff0c;并且比较了MoE和Dense模型的区别&#xff0c;今天我们继续来回顾一下MoE模型发展的历史和最新的发展现状。 从去年GPT-4发布至今&#xff0c;MoE…

AIGC重塑创意设计:不仅能带来新技术,更能引发新思考

随着科技的飞速发展&#xff0c;AIGC&#xff08;生成式人工智能&#xff09;已经逐渐成为创意设计领域的一股新势力。从影视制作到游戏设计&#xff0c;从平面广告到数字媒体&#xff0c;AIGC的影响力无处不在&#xff0c;它不仅带来了全新的技术手段&#xff0c;更在深层次上…

[SAP ABAP] 数据字典

ABAP数据字典是定义和管理数据库对象的工具 系统的所有全局数据类型以及数据库表结构等都需要在数据字典中创建和维护(数据字典中的对象对所有ABAP程序都是全局的) 通过数据字典&#xff0c;我们可以把数据库对象管理好&#xff0c;后续才能顺利的进行功能开发&#xff0c;SA…

安全为基、创新驱动,Soul App创始人张璐团队大力筑牢社交平台发展之基

随着技术的不断进步,AIGC在各个领域的应用日益广泛。在社交领域,AIGC社交为用户带来了更加高质量、个性化的社交体验,但同时也伴随着数据隐私泄露、网络诈骗和不良信息误导等风险。因此,社交平台在抓住AIGC技术发展机遇的同时,也要警惕技术所带来的风险。新型社交平台Soul App坚…

15_软件程序设计基础

目录 嵌入式软件开发原理 宿主机和目标机 交叉编译 交叉调试 嵌入式软件开发特点和挑战 开发工具 程序设计语言基本概念 解释和编译 常见程序设计语言 程序设计语言的基本成分 编译程序基本原理 嵌入式软件开发原理 宿主机和目标机 嵌入式软件开发不同于传统软件开…

IO-Link OD介绍

IO-Link OD&#xff08;On-request Data&#xff0c;按需数据&#xff09;是IO-Link通信中的一种重要数据类型&#xff0c;主要用于参数读写、指令交互、事件上传等动作。以下是关于IO-Link OD的结构、构成以及功能使用的详细说明&#xff1a; 结构与构成 定义&#xff1a;OD…

星火认知大模型Spark3.5 api调用 #AI夏令营 #Datawhale #夏令营

环境安装 pip install --upgrade -q spark_ai_python 官网注册 链接&#xff1a;https://console.xfyun.cn/app/myapp 官方会在报名成功后&#xff0c;立即发放星火大模型 Spark Max的 API 额度助力我们完成赛事&#xff0c; 200w Tokens 任你花&#xff01; 暑期有四场夏令营…

ADS基础教程23 - 有限元电磁仿真(FEM)可视化操作

EM介绍 一、引言二、FEM可视化操作流程1.打开可视化界面2.查看介质的网格3.设置网格颜色4.选择网格5.传感器选择6.编辑传感器7.选择频率8.动画 三、总结 一、引言 在ADS基础教程22中介绍了如何在ADS进行有限元电磁仿真&#xff08;FEM&#xff09;&#xff0c;本文将继续介绍F…

MSVCR120.DLL丢失的多种修复方法,助你快速解决dll问题

在日常生活和工作中&#xff0c;电脑已经成为我们不可或缺的工具。然而&#xff0c;在使用电脑的过程中&#xff0c;我们常常会遇到一些问题&#xff0c;其中之一就是电脑运行软件时提示找不到msvcr120.dll。如果该文件缺失或损坏&#xff0c;可能会导致依赖它的应用程序无法启…

解锁跨境电商新边界:Temu API接口深度解析引言

引言 在竞争激烈的跨境电商领域&#xff0c;高效、精准的数据获取成为商家制胜的关键。Temu&#xff08;拼多多跨境电商&#xff09;作为行业内的新秀&#xff0c;其API接口服务为商家提供了强大的数据交互能力&#xff0c;尤其是其获取商品详情的核心功能&#xff0c;更是为商…

2.3章节Python中的数值类型

1.整型数值 2.浮点型数值 3.复数   Python中的数值类型清晰且丰富&#xff0c;主要分为以下几种类型&#xff0c;每种类型都有其特定的用途和特性。 一、整型数值 1.定义&#xff1a;整数类型用于表示整数值&#xff0c;如1、-5、100等。 2.特点&#xff1a; Python 3中的…

【SGX系列教程】(三)Intel-SGX 官方示例分析(SampleCode)——SampleEnclave

文章目录 一. 引言二. README2.1 项目目的2.2 构建和执行示例代码的步骤2.3 配置参数解释2.4 配置文件分析2.5 启动令牌初始化 三. 重点代码分析3.1 App文件夹3.1.1 App/App.cpp3.1.2 App/Edger8rSyntax文件夹3.1.2.1 App/Edger8rSyntax/Arrays.cpp3.1.2.2 App/Edger8rSyntax/F…