LeetCode 11.盛最多水的容器

11.盛最多水的容器

题面:

给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。

找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

返回容器可以储存的最大水量。

说明:你不能倾斜容器。

示例:(建议参考原题面)

输入:[1,8,6,2,5,4,8,3,7]

输出:49

解释:(8, 7)情况下,容器能够容纳水的最大值为 49。

解析:
  • 对于这道题目,暴力就是两层循环,遍历所有的左右栏杆的情况,求解出其中的最大值。
  • 优化:我们可以进行的优化就是在答案的搜索空间内减少不必要的遍历验证。
  • 双指针的思想:它先令底边(right - left)最大,这样我们接下来的任何遍历操作都会使得底边变小。设当前我们遍历至step[i],此时的面积为min(height[left], height[right]) * (right - left),对于step[i+1]而言,我们可以进行的操作就是右移左边的栏杆(left++)与左移右边的栏杆(right--)。此时不论移动哪一边,底边都是变小,不妨设height[left] < height[right],ans = height[left] * (right - left),如果我们移动长的栏杆,会出现下面两种情况: c a s e 1 : h e i g h t [ − − r i g h t ] < h e i g h t [ l e f t ] → a r e a = h e i g h t [ − − r i g h t ] ∗ ( r i g h t − l e f t − 1 ) → a r e a < a n s c a s e 2 : h e i g h t [ − − r i g h t ] ≥ h e i g h t [ l e f t ] → a r e a = h e i g h t [ l e f t ] ∗ ( r i g h t − l e f t − 1 ) → a r e a < a n s case_1:height[--right] < height[left] \rightarrow area = height[--right] * (right - left - 1) \rightarrow area < ans\\case_2:height[--right] ≥ height[left] \rightarrow area = height[left] * (right - left - 1) \rightarrow area < ans case1:height[−−right]<height[left]→area=height[−−right]∗(right−left−1)→area<anscase2:height[−−right]≥height[left]→area=height[left]∗(right−left−1)→area<ans
  • 此后我们会发现,移动长的一条边的话,我们的搜索就是无意义的了,我们在以当前的面积为上界进行搜索,但是如果我们移动较短的一条边,就会有如下三种情况:
    c a s e 1 : h e i g h t [ + + l e f t ] > h e i g h t [ r i g h t ] → a r e a = h e i g h t [ r i g h t ] ∗ ( r i g h t − l e f t − 1 ) → a r e a ? a n s c a s e 2 : h e i g h t [ + + l e f t ] = h e i g h t [ r i g h t ] → a r e a = h e i g h t [ r i g h t ] ∗ ( r i g h t − l e f t − 1 ) → a r e a ? a n s c a s e 3 : h e i g h t [ + + l e f t ] < h e i g h t [ r i g h t ] → a r e a = h e i g h t [ + + l e f t ] ∗ ( r i g h t − l e f t − 1 ) → a r e a ? a n s case_1:height[++left] > height[right] \rightarrow area = height[right] * (right - left - 1) \rightarrow area ? ans\\case_2:height[++left] = height[right] \rightarrow area = height[right] * (right - left - 1) \rightarrow area ? ans\\case_3:height[++left] < height[right] \rightarrow area = height[++left] * (right - left - 1) \rightarrow area ? ans case1:height[++left]>height[right]→area=height[right]∗(right−left−1)→area?anscase2:height[++left]=height[right]→area=height[right]∗(right−left−1)→area?anscase3:height[++left]<height[right]→area=height[++left]∗(right−left−1)→area?ans
  • 此时我们可以发现,针对于移动较短的一边而言,此时可能存在比当前面积更大的情况,此时的搜索才是有意义的,而对于这道题目而言,就是基于这样实现的双指针,减少了很多无意义的搜索,将O(n ^ 2)的搜索空间使用O(n)的时间复杂度完成最大面积的搜索。
复杂度

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

Code
c++ 复制代码
// C++
class Solution {
public:
    int maxArea(vector<int>& height) {
        int left = 0;
        int right = height.size() - 1;
        int ans = min(height[left], height[right]) * (right - left);
        while (left < right)
        {
            if (height[left] <= height[right])
            {
                left++;
            } else
            {
                right--;
            }
            int temp = min(height[left], height[right]) * (right - left);
            ans = max(ans, temp);
        }
        return ans;
    }
};
python 复制代码
# Python
class Solution:
    def maxArea(self, height: List[int]) -> int:
        i, j = 0, len(height) - 1
        ans = min(height[i], height[j]) * (j - i)
        while i < j:
            if height[i] <= height[j]:
                i += 1
            else:
                j -= 1
            temp = min(height[i], height[j]) * (j - i)
            ans = max(temp, ans)
        return ans
rust 复制代码
// Rust
use std::cmp;

impl Solution {
    pub fn max_area(height: Vec<i32>) -> i32 {
        let mut left = 0;
        let mut right = height.len() - 1;
        let mut ans = cmp::min(height[left], height[right]) * (right - left) as i32;
        while left < right
        {
            if height[left] <= height[right]
            {
                left += 1;
            } else {
                right -= 1;
            }
            let temp = cmp::min(height[left], height[right]) * (right - left) as i32;
            ans = cmp::max(ans, temp);
        }
        ans
    }
}
相关推荐
reasonsummer10 小时前
【教学类-70-06】20260104“折纸-花纹镜”(花纹图案+上下打孔+无学号框)小2班第一次尝试
python·通义万相
阿猿收手吧!10 小时前
【C++】JSON核心数据结构解析及JSONCPP使用
数据结构·c++·json
leo__52010 小时前
基于C语言的FOC算法核心模块实现
c语言·算法·机器学习
kisshuan1239610 小时前
仪表盘数字识别-基于YOLOv8改进的EUCB-SC算法实现
算法·yolo
报错小能手10 小时前
线程池学习(二)线程池详解
c++·线程池
w-w0w-w10 小时前
C++泛型编程
开发语言·c++·算法
YJlio10 小时前
PsPing 学习笔记(14.8):常见错误与排障实战(超时、拒绝连接、权限问题)
开发语言·笔记·python·学习·django·pdf·pygame
-西门吹雪10 小时前
C++线程之内存模型
c++
LegendNoTitle10 小时前
Windows和Linux下Rust-init、Cargo下载慢的解决
开发语言·windows·rust
wjykp10 小时前
part6 PyTorch
人工智能·pytorch·python