题目
给你 n n n 个非负整数 a 1 , a 2 , . . . , a n a_1,a_2,...,a_n a1,a2,...,an,每个数代表坐标中的一个点 ( i , a i ) (i, a_i) (i,ai) 。在坐标内画 n n n 条垂直线,垂直线 i i i 的两个端点分别为 ( i , a i ) (i, a_i) (i,ai) 和 ( i , 0 ) (i, 0) (i,0)。找出其中的两条线,使得它们与 x x x 轴共同构成的容器可以容纳最多的水。
说明: 你不能倾斜容器,且 n n n 的值至少为 2。
图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。
示例:
输入:[1,8,6,2,5,4,8,3,7]
输出:49
思路1
直接暴力解决,定义一个双重循环,枚举所有的情况,计算出最大值,时间复杂度 ( O 2 ) (O^2) (O2);
代码1
public int maxArea(int[] height) {
int maxArea = 0;
for (int i=0; i<height.length-1; i++){
for (int j=i+1; j<height.length; j++){
maxArea = Math.max(maxArea, Math.min(height[i], height[j]) * (j-i));
}
}
return maxArea;
}
思路2
双指针法: 定义两个指针 i, j
,分别指向数组的首尾,指针移动的策略可以这样定义:由题目意思可知,容器容量大小由较矮的那根柱子决定,所以如果移动较高的那根柱子,容器的大小不可能增加,所以每次应该移动较矮的那根柱子;
代码2
public int maxArea(int[] height) {
int i = 0, j = height.length - 1, maxArea = 0;
while (i != j){
maxArea = Math.max(maxArea, Math.min(height[i], height[j]) * (j - i));
if (height[i] < height[j])
i++;
else
j--;
}
return maxArea;
}