微软和谷歌的几个大佬组织了一个面试刷题群,可以加管理员VX:sxxzs3998(备注掘金),进群参与讨论和直播
1. 问题
给定 个非负整数 ,每个数代表坐标中的一个点 。在坐标内画 条垂直线,垂直线 的两个端点分别为 和 。找出其中的两条线,使得它们与 轴共同构成的容器可以容纳最多的水。
说明:你不能倾斜容器,且 的值至少为 2。
示例1
输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]
输出:6
解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。
图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。
示例 2:
输入:height = [4,2,0,3,2,5]
输出:9
2. 解析
2.1 方法一:暴力法
首先最容易想到的就是暴力去搜索,检验每两个匹配形成的面积有多大,代码如下:
public static int maxArea(int[] height) {
int max = 0;
int current;
for(int i =0; i<height.length - 1; i++){
for(int j = i+ 1; j< height.length; j++){
current = (j-i) * Math.min(height[i],height[j]);
max = Math.max(max,current);
}
}
return max;
}
复杂度分析:
- 时间复杂度:O(n²)。
- 空间复杂度:O(1)。
2.2 方法二:双指针法
两线段之间形成的区域总是会受到其中较短那条长度的限制。我们使用两个指针,一个放在开始,一个置于末尾。 在每一步中,我们会计算指针所指向的两条线段形成的区域面积,并将指向较短线段的指针向较长线段那端移动一步。如下:
public static int maxArea2(int[] height) {
int max = 0;
int current = 0;
int left = 0;
int right = height.length-1;
while(left < right){
current = (right - left) * Math.min(height[left],height[right]);
max = Math.max(max,current);
if(height[left] < height[right]){
left ++ ;
}else{
right -- ;
}
}
return max;
}
复杂度分析:
- 时间复杂度:O(n)。
- 空间复杂度:O(1)。
微软和谷歌的几个大佬组织了一个面试刷题群,可以加管理员VX:sxxzs3998(备注掘金),进群参与讨论和直播