LeetCode 42. 接雨水(C语言详解)——双指针经典解法

一、题目描述

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

示例 1:

复制代码
输入:
height = [0,1,0,2,1,0,1,3,2,1,2,1]

输出:
6

解释:

该高度图在下雨后可以接到 6 个单位的雨水

示例 2:

复制代码
输入:
height = [4,2,0,3,2,5]
输出:
9

二、问题分析

每个位置能接多少雨水,取决于:

复制代码
min(左侧最高柱子, 右侧最高柱子) - 当前柱子高度

例如:

复制代码
      █
  █   █
█ █ █ █

如果某一位置:

复制代码
左边最高 = 4
右边最高 = 3
当前高度 = 1

那么该位置可以接的水为:

复制代码
min(4,3) - 1 = 2

三、双指针解法(最优解)

为了高效计算,我们可以使用 双指针 + 动态维护左右最大值

定义两个指针:

复制代码
left  = 0
right = n - 1

同时维护:

复制代码
leftMax  = 左侧最大高度
rightMax = 右侧最大高度

核心思想:

  • 如果 height[left] < height[right]

    • 说明左侧是短板

    • leftMax 计算水量

  • 否则

    • 右侧是短板

    • rightMax 计算水量

因为水位始终取决于 较矮的一侧


四、算法流程

  1. 初始化指针

    left = 0
    right = n - 1

  2. 记录左右最大高度

    leftMax = 0
    rightMax = 0

  3. 遍历数组

    while(left < right)

  4. 判断左右高度关系

若:

复制代码
height[left] < height[right]

更新左侧:

复制代码
leftMax = max(leftMax, height[left])

water += leftMax - height[left]

否则更新右侧:

复制代码
rightMax = max(rightMax, height[right])

water += rightMax - height[right]
  1. 移动指针

直到 left >= right


五、C语言代码实现

复制代码
int trap(int* height, int heightSize) {

    int left = 0;
    int right = heightSize - 1;

    int leftMax = 0;
    int rightMax = 0;

    int water = 0;

    while (left < right) {

        if (height[left] < height[right]) {

            if (height[left] >= leftMax)
                leftMax = height[left];
            else
                water += leftMax - height[left];

            left++;

        } else {

            if (height[right] >= rightMax)
                rightMax = height[right];
            else
                water += rightMax - height[right];

            right--;
        }
    }

    return water;
}

六、复杂度分析

复杂度 说明
时间复杂度 O(n)
空间复杂度 O(1)

只遍历一次数组,并且没有使用额外数组空间。


七、示例分析

输入:

复制代码
height = [0,1,0,2,1,0,1,3,2,1,2,1]

计算过程会累计得到:

复制代码
1 + 1 + 2 + 1 + 1 = 6

最终结果:

复制代码
6

八、总结

本题是 经典的双指针问题,核心思想是:

复制代码
水位 = min(左侧最高柱子, 右侧最高柱子)

通过双指针不断向中间收缩,并动态维护 leftMaxrightMax,就可以在 O(n) 时间内计算出所有雨水量。

这种方法相比使用 leftMax[]rightMax[] 数组的解法,空间复杂度从 O(n) 降到了 O(1),是面试中更推荐的写法。

相关推荐
查古穆5 小时前
栈-有效的括号
java·数据结构·算法
再一次等风来5 小时前
近场声全息(NAH)仿真实现:从阵列实值信号到波数域重建
算法·matlab·信号处理·近场声全息·nah
汀、人工智能5 小时前
16 - 高级特性
数据结构·算法·数据库架构·图论·16 - 高级特性
大熊背5 小时前
利用ISP离线模式进行分块LSC校正的方法
人工智能·算法·机器学习
XWalnut5 小时前
LeetCode刷题 day4
算法·leetcode·职场和发展
蒸汽求职5 小时前
机器人软件工程(Robotics SDE):特斯拉Optimus落地引发的嵌入式C++与感知算法人才抢夺战
大数据·c++·算法·职场和发展·机器人·求职招聘·ai-native
charlee445 小时前
最小二乘问题详解17:SFM仿真数据生成
c++·计算机视觉·sfm·数字摄影测量·无人机航测
Tanecious.6 小时前
蓝桥杯备赛:Day4-P9749 公路
c++·蓝桥杯
AI成长日志6 小时前
【笔面试算法学习专栏】双指针专题·简单难度两题精讲:167.两数之和II、283.移动零
学习·算法·面试
旖-旎6 小时前
分治(库存管理|||)(4)
c++·算法·leetcode·排序算法·快速选择算法