[贪心|双指针] leetcode 11 盛最多水的容器

[贪心|双指针] leetcode 11 盛最多水的容器

1.题目

题目链接
给你 n 个非负整数 a1,a2,…,an,每个数代表坐标中的一个点 (i, ai) 。在坐标内画 n 条垂直线,垂直线 i 的两个端点分别为 (i, ai) 和 (i, 0)。找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。
说明: 你不能倾斜容器,且 n 的值至少为 2。
示例:

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

在这里插入图片描述
图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。

2.分析

2.1.直观解法

本题相当于模拟"木桶效应",我们只需要穷举每种可能的"容器左端"、“容器右端”,然后比较各个容器的盛水量即可:

int maxArea(vector<int>& height) {
    int n = height.size();
    int ans = -1;
    //容器左端
    for(int l = 0; l < n; l++) {
    	//容器右端
        for(int r = l + 1; r < n; r++) {
        	//比较最大盛水量与当前盛水量
            ans = max(ans, min(height[l], height[r]) * (r - l));
        }
    }
    return ans;
}

这种做法的时间复杂度为O(n2),在提交时会直接TLE。有没有更好的解法呢?

2.2.双指针初探

在先前的直观解法中,两个指针为lrl表示容器的左边界,r表示容器的右边界,容器的容积为min(height[l], height[r]) * (r - l)。
现在,我们仍然保留l、r的定义,只不过改变l、r的初始值,有:

int l = 0, r = n - 1;
while(l < r) {
    if(???) {
        l++;
    }else {
        r--;
    }
}

将l初始为数组的最左端,r初始为数组的最右端,遍历各容器的过程则变成了"容器宽度逐渐减小的过程"。

2.3.有规则的搜索

给定了l为0,r为n-1,接下来要解决问题便是l在什么情况下增大r在什么情况下减小
对于某一时间点,我们拥有的信息只有height[l]与height[r]。考虑任意一种情况,比如height[l] < height[r];此时能执行的操作也只有两种可能:l右移或者是r左移。

  • l右移:若height[l + 1] > height[l],由于height[l]是短板,“短板变长”,因此有可能更新得到一个更优解;反之,“短板变短”,不可能得到一个更优解。
  • r左移:由于长板并不会影响容器的容积,因此无论height[r - 1]大于还是小于height[r],都不可能得到一个更优解。

因此,在height[l] < height[r]时,l应该右移;反之,r应该左移。

3.代码

class Solution {
public:
    int maxArea(vector<int>& height) {
        int n = height.size();
        int l = 0, r = n - 1;
        int ans = -1;
        while(l < r) {
            ans = max(ans, min(height[l], height[r]) * (r - l));
            if(height[l] < height[r]) {
                l++;
            }else {
                r--;
            }
        }
        return ans;
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值