zoukankan      html  css  js  c++  java
  • 【转】Twitter算法面试题详解(Java实现)

    原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、作者信息和本声明。否则将追究法律责任。http://androidguy.blog.51cto.com/974126/1319659

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

                 图1

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

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

    代码中的测试用例的结果是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

    有其他算法的(语言不限)欢迎跟帖......

    博客地址: http://www.cnblogs.com/dwf07223,本文以学习、研究和分享为主,欢迎转载,转载请务必保留此出处。若本博文中有不妥或者错误处请不吝赐教。

  • 相关阅读:
    selenium
    python第三方模块的安装
    程序员学习网站
    python 数据较大 性能分析
    linux ~/ 和 /
    VMWare虚拟机 window文件传递
    vi命令
    os.system
    win10系统进入BIOS
    pyinstaller将python脚本生成exe
  • 原文地址:https://www.cnblogs.com/dwf07223/p/3418005.html
Copyright © 2011-2022 走看看