盛最多水的容器:双指针法的巧妙运用(leetcode 11)

问题描述

给定一个长度为 n 的整数数组 height,有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i])。找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

我的思考历程

最初看到这个问题,我直觉上想到的是双指针法。设置左右两个指针,分别指向数组的开头和结尾,然后逐步向中间移动。但关键问题是:应该移动哪个指针?

第一次尝试:移动较高的指针

我最初的思路是:"让高的柱子先走,因为它可能找到更高的搭档"。于是写出了这样的代码:

cpp

复制代码
// 初始错误思路
if (height[left] > height[right]) {
    h = min(height[left], height[--right]);
} else {
    h = min(height[++left], height[right]);
}

但测试后发现结果不对!这让我陷入了深深的思考...

突破时刻:为什么应该移动较矮的指针?

经过1个小时的思考和数学推导,我终于明白了其中的奥秘:

数学证明

设当前状态:容量 = min(h_left, h_right) × width

如果移动较高的指针

  • 新宽度 = width - 1

  • 新高度 ≤ min(h_left, h_right)

  • ∴ 新容量 ≤ min(h_left, h_right) × (width - 1) < 原容量

如果移动较矮的指针

  • 新宽度 = width - 1

  • 新高度可能 > min(h_left, h_right)

  • ∴ 新容量可能增加!

直观理解

想象两个柱子,一个矮一个高:

  • 移动高的柱子:容器的高度上限被矮柱子限制,宽度减少,容量必然减少

  • 移动矮的柱子:虽然宽度减少,但可能找到更高的柱子,从而增加高度补偿宽度的损失

最终解决方案

cpp

复制代码
class Solution {
public:
    int maxArea(vector<int>& height) {
        int maxArea = 0;
        int left = 0;
        int right = height.size() - 1;
        
        while (left < right) {
            int currentHeight = min(height[left], height[right]);
            int currentWidth = right - left;
            maxArea = max(maxArea, currentHeight * currentWidth);
            
            // 关键 insight:移动较矮的指针
            if (height[left] < height[right]) {
                left++;
            } else {
                right--;
            }
        }
        
        return maxArea;
    }
};

算法分析

  • 时间复杂度:O(n),每个元素最多被访问一次

  • 空间复杂度:O(1),只使用常数额外空间

  • 正确性保证:每次移动都排除了不可能产生更大容量的情况

相关推荐
YSRM2 分钟前
Leetcode+Java+图论II
java·leetcode·图论
csdn_aspnet10 分钟前
C++ 圆台体积和表面积计算程序(Program for Volume and Surface area of Frustum of Cone)
c++
.NET修仙日记31 分钟前
.NET WinForms + WPF 综合学习路线:从传统到现代的.NET桌面开发
学习·c#·.net·wpf·.net core·winforms
crary,记忆1 小时前
MFE: React + Angular 混合demo
前端·javascript·学习·react.js·angular·angular.js
我狸才不是赔钱货1 小时前
AI大模型“战国策”:主流LLM平台简单介绍
c++·人工智能·程序人生·github·llama
无限进步_1 小时前
【C语言】在矩阵中高效查找数字的算法解析
c语言·开发语言·数据结构·c++·其他·算法·矩阵
Yupureki2 小时前
从零开始的C++学习生活 11:二叉搜索树全面解析
c语言·数据结构·c++·学习·visual studio
电子云与长程纠缠2 小时前
Blender入门学习02
学习·blender
再睡一夏就好2 小时前
【C++闯关笔记】STL:deque与priority_queue的学习和使用
java·数据结构·c++·笔记·学习·
天选之女wow2 小时前
【代码随想录算法训练营——Day43(Day42周日休息)】动态规划——300.最长递增子序列、674.最长连续递增序列、718.最长重复子数组
算法·leetcode·动态规划