【微软算法面试高频题】接雨水

2021/7/18 9:35:51

本文主要是介绍【微软算法面试高频题】接雨水,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!

微软和谷歌的几个大佬组织了一个面试刷题群,可以加管理员VX:sxxzs3998(备注CSDN),进群参与讨论和直播

1. 问题

给定 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。

image.png

示例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 方法二:双指针法

两线段之间形成的区域总是会受到其中较短那条长度的限制。我们使用两个指针,一个放在开始,一个置于末尾。 在每一步中,我们会计算指针所指向的两条线段形成的区域面积,并将指向较短线段的指针向较长线段那端移动一步。如下:

image.png

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(备注CSDN),进群参与讨论和直播



这篇关于【微软算法面试高频题】接雨水的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!


扫一扫关注最新编程教程