双指针巧解LeetCode接雨水难题

LeetCode 42 题「接雨水」是数组类难题的经典代表,考察对动态规划 / 双指针 的理解。本文将拆解双指针法的核心思路,并给出简洁高效的代码实现。

一、题目理解

给定 n 个非负整数表示 "宽度为 1 的柱子高度",计算这些柱子排列后能接住的雨水量。

  • 示例:输入[0,1,0,2,1,0,1,3,2,1,2,1],输出6(蓝色区域为雨水)。

二、核心思路:双指针法(O (n) 时间 + O (1) 空间)

接雨水的本质是:每个位置能接的雨水量 = 该位置左右两侧的 "最高柱子高度的较小值" - 自身高度

双指针法通过左右指针向中间收缩,避免了动态规划的额外空间开销,核心逻辑如下:

  1. 定义left(左指针,初始 0)、right(右指针,初始数组末尾);
  2. 定义pleft(左指针左侧的最高高度)、pright(右指针右侧的最高高度);
  3. 比较pleftpright
    • pleft < pright:左指针位置的接水量由pleft决定(右侧有更高的柱子兜底),计算当前位置接水量后左指针右移;
    • pleft >= pright:右指针位置的接水量由pright决定(左侧有更高的柱子兜底),计算当前位置接水量后右指针左移;
  4. 每次移动指针时,更新对应的 "最高高度"。

三、代码实现(C++)

cpp

复制代码
class Solution {
public:
    int trap(vector<int>& height) {
        if (height.empty()) return 0; // 空数组直接返回0
        int pleft = 0, pright = 0;    // 左右侧的最高高度
        int left = 0;                 // 左指针
        int right = height.size() - 1;// 右指针
        int sum = 0;                  // 总接水量
        
        while (left < right) {
            // 更新左侧最高高度
            pleft = max(pleft, height[left]);
            // 更新右侧最高高度
            pright = max(pright, height[right]);
            
            // 计算当前位置接水量,并移动指针
            if (pleft < pright) {
                sum += pleft - height[left];
                left++;
            } else {
                sum += pright - height[right];
                right--;
            }
        }
        return sum;
    }
};

四、代码解析

  • 初始条件:空数组直接返回 0,避免边界错误;
  • 指针移动逻辑:始终让 "较矮的一侧指针" 移动,保证当前计算的接水量是有效的(因为另一侧有更高的柱子能接住水);
  • 空间优化:仅用常数级变量存储状态,相比动态规划的 O (n) 空间,双指针法空间复杂度为 O (1)。

五、复杂度分析

  • 时间复杂度:O (n)(仅遍历数组一次);
  • 空间复杂度:O (1)(无额外数组开销)。

双指针法是「接雨水」的最优解法之一,既高效又简洁,核心是抓住 "接水量由两侧较矮的最高柱决定" 这一规律。

相关推荐
alvin_20055 小时前
python之OpenGL应用(二)Hello Triangle
python·opengl
机器视觉的发动机5 小时前
AI算力中心的能耗挑战与未来破局之路
开发语言·人工智能·自动化·视觉检测·机器视觉
铁蛋AI编程实战5 小时前
通义千问 3.5 Turbo GGUF 量化版本地部署教程:4G 显存即可运行,数据永不泄露
java·人工智能·python
HyperAI超神经5 小时前
在线教程|DeepSeek-OCR 2公式/表格解析同步改善,以低视觉token成本实现近4%的性能跃迁
开发语言·人工智能·深度学习·神经网络·机器学习·ocr·创业创新
晚霞的不甘5 小时前
CANN 编译器深度解析:UB、L1 与 Global Memory 的协同调度机制
java·后端·spring·架构·音视频
jiang_changsheng5 小时前
RTX 2080 Ti魔改22GB显卡的最优解ComfyUI教程
python·comfyui
SunnyDays10115 小时前
使用 Java 冻结 Excel 行和列:完整指南
java·冻结excel行和列
R_.L5 小时前
【QT】常用控件(按钮类控件、显示类控件、输入类控件、多元素控件、容器类控件、布局管理器)
开发语言·qt
Zach_yuan5 小时前
自定义协议:实现网络计算器
linux·服务器·开发语言·网络
摇滚侠5 小时前
在 SpringBoot 项目中,开发工具使用 IDEA,.idea 目录下的文件需要提交吗
java·spring boot·intellij-idea