Twitter算法面试题详解(Java实现)

来源:互联网 发布:快手段子制作软件 编辑:程序博客网 时间:2024/06/13 21:44

最近在网上看到一道Twitter的算法面试题,网上已经有人给出了答案,不过可能有些人没太看明白(我也未验证是否正确),现在给出一个比较好理解的答案。先看一下题目。

图1

先看看这个图。可以将方块看做砖。题干很简单,问最多能放多少水。例如,图2就是图1可放的最多水(蓝色部分),如果将一块砖看做1的话,图2就是能放10个单位的水。


图2

再看个例子

图3

图3可以放17个单位的水。

上面每一个图的砖墙用int数组表示,每一个数组元素表示每一列砖墙的砖数(高度),例如,图3用数组表示就是int[] wallHeights = new int[]{2, 5, 1, 3, 1, 2, 1, 7, 7, 6};

这里某人给出了python的算法点击打开链接,不过有人说有问题,有python环境的可以验证。现在给出我的Java算法。


算法原理

其实很简单,我的算法并不是累加的,而是用的减法,先用图3为例。只需要找到所有墙中最高的,然后再找出第二高的。如果两堵墙紧邻者,就忽略它,否则算一下如果墙之间没有任何其他的砖的情况下可以有多少水(只是一个乘法而已),然后扫描两堵墙之间有多少块砖,减去这个砖数就可以了。最后用递归处理。将两堵墙两侧到各自的左右边界再重新进行前面的操作(递归处理)。直到无墙可处理。 用递归方法很容易理解。下面看一下算法的详细代码。

[java] view plaincopy
  1. public class Test  
  2. {  
  3.     static int result = 0;  //  最终结果  
  4.     static int[] wallHeights = new int[]  
  5.     {1,6,1,2,3,4,100,1,9};  //  表示所有的墙的高度  
  6.   
  7.     public static void process(int start, int end)  
  8.     {  
  9.         //  first:start和end之间最高的墙  
  10.         //  second:start和end之间第二高的墙  
  11.         int first = 0, second = 0;  
  12.         //  firstIndex:第一高的墙在wallHeights中的索引  
  13.         //  secondIndex:第二高的墙在wallHeights中的索引  
  14.         int firstIndex = 0, secondIndex = 0;  
  15.         //  两堵墙必须至少有一堵墙的距离  
  16.         if (end - start <= 1)  
  17.             return;  
  18.         //  开始获取第一高和第二高墙的砖数  
  19.         for (int i = start; i <= end; i++)  
  20.         {  
  21.             if (wallHeights[i] > first)  
  22.             {  
  23.                 second = first;  
  24.                 secondIndex = firstIndex;  
  25.                 first = wallHeights[i];  
  26.                 firstIndex = i;  
  27.             }  
  28.             else if (wallHeights[i] > second)  
  29.             {  
  30.                 second = wallHeights[i];  
  31.                 secondIndex = i;  
  32.             }  
  33.         }  
  34.   
  35.         //  获取左侧墙的索引  
  36.         int startIndex = Math.min(firstIndex, secondIndex);  
  37.         //  获取右侧墙的索引  
  38.         int endIndex = Math.max(firstIndex, secondIndex);  
  39.         //  计算距离  
  40.         int distance = endIndex - startIndex;  
  41.         //  如果第一高的墙和第二高的墙之间至少有一堵墙,那么开始计算这两堵墙之间可以放多少个单位的水  
  42.         if (distance > 1)  
  43.         {  
  44.             result = result + (distance - 1) * second;  
  45.             //  减去这两堵墙之间的砖数  
  46.             for (int i = startIndex + 1; i < endIndex; i++)  
  47.             {  
  48.                 result -= wallHeights[i];  
  49.             }  
  50.               
  51.         }  
  52.         //  开始递归处理左侧墙距离开始位置能放多少水  
  53.         process(start, startIndex);  
  54.         //  开始递归处理右侧墙距离结束位置能放多少水  
  55.         process(endIndex, end);  
  56.     }  
  57.   
  58.     public static void main(String[] args)  
  59.     {  
  60.         process(0, wallHeights.length - 1);  
  61.         System.out.println(result);  
  62.   
  63.     }  
  64.   
  65. }  
代码中的测试用例的结果是22。下面是几组测试用例。


[2,5,1,2,3,4,7,7,6]   结果:10
[2,5,1,3,1,2,1,7,7,6]  结果:17
[6,1,4,6,7,5,1,6,4]   结果:13
[9,6,1,2,3,4,50,1,9]  结果:37

0 0