我们在处理一道编程面试题的时候,通常除了注意代码规范以外,千万要记得自己心中模拟一个单元测试。主要通过三方面来处理。

  • 功能性测试
  • 边界值测试
  • 负面性测试

不管如何,一定要保证自己代码考虑的全面,而不要简单地猜想用户的输入一定是正确的,只是去实现功能。通常你编写一个能接受住考验的代码,会让面试官对你刮目相看,你可以不厉害,但已经充分说明了你的靠谱。

今天我们的面试题目是:

面试题:尝试实现 Java 的 Math.pow(double base,int exponent) 函数算法,计算 base 的 exponent 次方,不得使用库函数,同时不需要考虑大数问题。

面试题来源于《剑指 Offer》第 11 题,数字的整数次方。

不要介意 Java 真正的方法是 Math.pow(double var1,double var2)。

由于不需要考虑大数问题,不少小伙伴心中暗自窃喜,这题目也太简单了,给我撞上了,运气真好,于是直接写出下面的代码:

public class Test11 {

    private static double power(double base, int exponent) {
double result = 1.0;
for (int i = 0; i < exponent; i++) {
result *= base;
}
return result;
} public static void main(String[] args) {
System.out.println(power(2, 2));
System.out.println(power(2, 4));
System.out.println(power(3, 1));
System.out.println(power(3, 0));
}
}

写的快自然是好事,如果正确的话会被面试官认为是思维敏捷。但如果考虑不周的话,恐怕就极容易被面试官认为是不靠谱的人了。在技术能力和靠谱度之间,大多数面试官更青睐于靠谱度。

我们上面确实做到了功能测试,但面试官可能会直接提示我们,假设我们的 exponent 输入一个负值,能得到正确值么?

跟着自己的代码走一遍,终于意识到了这个问题,当 exponent 为负数的时候,循环根本就进不去,无论输入的负数是什么,都会返回 1.0,这显然是不正确的算法。

我们在数学中学过,给一个数值上负数次方,相当于给这个数值上整数次方再求倒数。

意识到这点,我们修正一下代码。

public class Test11 {

    private static double power(double base, int exponent) {
// 因为除了 0 以外,任何数值的 0 次方都为 1,所以我们默认为 1.0;
// 0 的 0 次方,在数学书是没有意义的,为了贴切,我们也默认为 1.0
double result = 1.0;
// 处理负数次方情况
boolean isNegetive = false;
if (exponent < 0) {
isNegetive = true;
exponent = -exponent;
}
for (int i = 0; i < exponent; i++) {
result *= base;
}
if (isNegetive)
return 1 / result;
return result;
} public static void main(String[] args) {
System.out.println(power(2, 2));
System.out.println(power(2, 4));
System.out.println(power(3, 1));
System.out.println(power(3, -1));
}
}

我们在代码中增加了一个判断是否为负数的 isNegetive 变量,当为负数的时候,我们就置为 true,并计算它的绝对值次幂,最后返回结果的时候返回它的倒数。

面试官看到这样的代码,可能就有点按捺不住内心的怒火了,不过由于你此前一直面试回答的较好,也打算再给你点机会,面试官提示你,当 base 传入 0,exponent 传入负数,会怎样?

瞬间发现了自己的问题,这不是犯了数学最常见的问题,给 0 求倒数么?

虽然 Java 的 Math.pow() 方法也存在这个问题,但我们这里忽略不计。

于是马上更新代码。

public class Test11 {

    private static double power(double base, int exponent) {
// 因为除了 0 以外,任何数值的 0 次方都为 1,所以我们默认为 1.0;
// 0 的 0 次方,在数学书是没有意义的,为了贴切,我们也默认为 1.0
double result = 1.0;
// 处理底数为 0 的情况,底数为 0 其他任意次方结果都应该是 0
if (base == 0)
return 0.0;
// 处理负数次方情况
boolean isNegetive = false;
if (exponent < 0) {
isNegetive = true;
exponent = -exponent;
}
for (int i = 0; i < exponent; i++) {
result *= base;
}
if (isNegetive)
return 1 / result;
return result;
} public static void main(String[] args) {
System.out.println(power(2, 2));
System.out.println(power(2, 4));
System.out.println(power(3, 1));
System.out.println(power(0, -1));
}
}

有了上一次的经验,这次并不敢直接上交代码了,而是认真检查边界值和各种情况。检查 1 遍,2 遍,均没有发现问题,提交代码。

计算机表示小数均有误差,这个在 Python 中尤其严重,但经数次测试,《剑指 Offer》中讲的双精度误差问题似乎在 Java 的 == 运算符中并不存在。如有问题,欢迎指正。

上面的代码基本还算整,健壮性也还不错,但面试官可能还想问问有没有更加优秀的算法。

仔细查看,确实似乎是有办法优化的,比如我们要求 power(2,16) 的值,我们只需要先求出 2 的 8 次方,再平方就可以了;以此类推,我们计算 2 的 8 次方的时候,可以先计算 2 的 4 次方,然后再做平方运算.....妙哉妙哉!

需要注意的是,如果我们的幂数为奇数的话,我们需要在最后再乘一次我们的底数。

我们尝试修改代码如下:

public class Test11 {
private static double power(double base, int exponent) {
// 因为除了 0 以外,任何数值的 0 次方都为 1,所以我们默认为 1.0;
// 0 的 0 次方,在数学书是没有意义的,为了贴切,我们也默认为 1.0
double result = 1.0;
// 处理底数为 0 的情况,底数为 0 其他任意次方结果都应该是 0
if (base == 0)
return 0.0;
// 处理负数次方情况
boolean isNegetive = false;
if (exponent < 0) {
isNegetive = true;
exponent = -exponent;
}
result = getTheResult(base, exponent);
if (isNegetive)
return 1 / result;
return result;
} private static double getTheResult(double base, int exponent) {
// 如果指数为0,返回1
if (exponent == 0) {
return 1;
}
// 指数为1,返回底数
if (exponent == 1) {
return base;
}
// 递归求一半的值
double result = getTheResult(base, exponent >> 1);
// 求最终值,如果是奇数,还要乘一次底数
result *= result;
if ((exponent & 0x1) == 1) {
result *= base;
}
return result; } public static void main(String[] args) {
System.out.println(power(2, 2));
System.out.println(power(2, 4));
System.out.println(power(3, -1));
System.out.println(power(0.1, 2));
}
}

完美解决。

在提交代码的时候,还可以主动提示面试官,我们在上面用右移运算符代替了除以 2,用位与运算符代替了求余运算符 % 来判断是一个奇数还是一个偶数。让他知道我们对编程的细节真的很重视,这大概也就是细节决定成败吧。一两个细节的打动说不定就让面试官下定决心给我们发放 Offer 了。

位运算的效率比乘除法及求余运算的效率要高的多。

因为移位指令占 2 个机器周期,而乘除法指令占 4 个机器周期。从硬件上看,移位对硬件更容易实现,所以我们更优先用移位。

好了,今天我们的面试精讲就到这里,我们明天再见!

面试 5:手写 Java 的 pow() 实现的更多相关文章

  1. 阿里第二轮面试:手写Java二叉树

    阿里面试 现在很多公司在招聘开发岗位的时候,都会事先在招聘信息中注明面试者应当具备的知识技能,而且在面试的过程中,有部分对于技能掌握程度有严格要求的公司还会要求面试者手写代码,这个环节很考验面试者的基 ...

  2. 手写JAVA虚拟机(二)——实现java命令行

    查看手写JAVA虚拟机系列可以进我的博客园主页查看. 我们知道,我们编译.java并运行.class文件时,需要一些java命令,如最简单的helloworld程序. 这里的程序最好不要加包名,因为加 ...

  3. 手写JAVA虚拟机(三)——搜索class文件并读出内容

    查看手写JAVA虚拟机系列可以进我的博客园主页查看. 前面我们介绍了准备工作以及命令行的编写.既然我们的任务实现命令行中的java命令,同时我们知道java命令是将class文件(字节码)转换成机器码 ...

  4. 6 手写Java LinkedHashMap 核心源码

    概述 LinkedHashMap是Java中常用的数据结构之一,安卓中的LruCache缓存,底层使用的就是LinkedHashMap,LRU(Least Recently Used)算法,即最近最少 ...

  5. 3 手写Java HashMap核心源码

    手写Java HashMap核心源码 上一章手写LinkedList核心源码,本章我们来手写Java HashMap的核心源码. 我们来先了解一下HashMap的原理.HashMap 字面意思 has ...

  6. java面试:手写代码

    二分查找法. /** * 二分查找法:给定一组有序的数组,每次都从一半中查找.直到找到要求的数据. * 主要是得找到下标的表示方法. */ public class BinaryFind { /** ...

  7. java面试之手写单例模式

    为什么要有单例模式 实际编程应用场景中,有一些对象其实我们只需要一个,比如线程池对象.缓存.系统全局配置对象等.这样可以就保证一个在全局使用的类不被频繁地创建与销毁,节省系统资源. 实现单例模式的几个 ...

  8. 2 手写Java LinkedList核心源码

    上一章我们手写了ArrayList的核心源码,ArrayList底层是用了一个数组来保存数据,数组保存数据的优点就是查找效率高,但是删除效率特别低,最坏的情况下需要移动所有的元素.在查找需求比较重要的 ...

  9. 1 手写Java ArrayList核心源码

    手写ArrayList核心源码 ArrayList是Java中常用的数据结构,不光有ArrayList,还有LinkedList,HashMap,LinkedHashMap,HashSet,Queue ...

随机推荐

  1. (后端)SQL SERVER 字符串按数字排序

    应用于B1-1,B1-2,B10-1,B11-1 sqlserver肯定不能按照字符串进行排序,需要进行处理一番: select CONVERT(varchar, LEFT(code,1)),conv ...

  2. JS 调试中常见的报错的解决办法

    报错:Uncaught SyntaxError: Unexpected token o in JSON at position 1 at JSON.parse (<anonymous>) ...

  3. js获取地址栏中的数据

    window.location.href:设置或获取整个 URL 为字符串window.location.pathname:设置或获取对象指定的文件名或路径window.location.search ...

  4. c/c++ 标准库 map set 删除

    标准库 map set 删除 删除操作 有map如下: map<int, size_t> cnt{{2,22}, {3,33}, {1,11}, {4,44}; 删除方法: 删除操作种类 ...

  5. 常见 User-Agent 大全(自己在用)

    分享几个常见的User-Agent吧,复制粘贴过来的,谢谢原创. window.navigator.userAgent 1) Chrome Win7: Mozilla/5.0 (Windows NT ...

  6. 常用判断重复记录的SQL语句

    1.查找表中多余的重复记录,重复记录是根据单个字段(peopleId)来判断select * from people where peopleId in (select   peopleId  fro ...

  7. C# -- 抽象类与抽象方法

    C#: 抽象类与抽象方法 1.代码 class Program { static void Main(string[] args) { ; i < ; i++) { == ) { Storage ...

  8. puppet 横向扩展(一)

    目录 1. 概述 2. 实验环境 3. 实验步骤 3.1. 创建puppetmaster的rack环境 3.2. 配置文件设置 3.3. 补充说明 3.4. 测试配置结果 3.4.1. 默认的负载均衡 ...

  9. 快速排序 O(n logn) 堆排序 O(n logn) 归并排序 O(n logn)

    NB三人组 快速排序 思路" 取一个元素P (第一个元素), 使元素归位 列表被P 分成两部分,左边都比P小,右边比P大; 递归完成排序. 问题 如果是已经排序好的 倒叙 列表 则会 递归深 ...

  10. Ubuntu16.04 下 hadoop的安装与配置(伪分布式环境)

    一.准备 1.1创建hadoop用户 $ sudo useradd -m hadoop -s /bin/bash #创建hadoop用户,并使用/bin/bash作为shell $ sudo pass ...