[贪心|双指针] 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.双指针初探
在先前的直观解法中,两个指针为l、r,l表示容器的左边界,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;
}
};