面试 5:手写 Java 的 pow() 实现
我们在处理一道编程面试题的时候,通常除了注意代码规范以外,千万要记得自己心中模拟一个单元测试。主要通过三方面来处理。
- 功能性测试
- 边界值测试
- 负面性测试
不管如何,一定要保证自己代码考虑的全面,而不要简单地猜想用户的输入一定是正确的,只是去实现功能。通常你编写一个能接受住考验的代码,会让面试官对你刮目相看,你可以不厉害,但已经充分说明了你的靠谱。
今天我们的面试题目是:
面试题:尝试实现 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() 实现的更多相关文章
- 阿里第二轮面试:手写Java二叉树
阿里面试 现在很多公司在招聘开发岗位的时候,都会事先在招聘信息中注明面试者应当具备的知识技能,而且在面试的过程中,有部分对于技能掌握程度有严格要求的公司还会要求面试者手写代码,这个环节很考验面试者的基 ...
- 手写JAVA虚拟机(二)——实现java命令行
查看手写JAVA虚拟机系列可以进我的博客园主页查看. 我们知道,我们编译.java并运行.class文件时,需要一些java命令,如最简单的helloworld程序. 这里的程序最好不要加包名,因为加 ...
- 手写JAVA虚拟机(三)——搜索class文件并读出内容
查看手写JAVA虚拟机系列可以进我的博客园主页查看. 前面我们介绍了准备工作以及命令行的编写.既然我们的任务实现命令行中的java命令,同时我们知道java命令是将class文件(字节码)转换成机器码 ...
- 6 手写Java LinkedHashMap 核心源码
概述 LinkedHashMap是Java中常用的数据结构之一,安卓中的LruCache缓存,底层使用的就是LinkedHashMap,LRU(Least Recently Used)算法,即最近最少 ...
- 3 手写Java HashMap核心源码
手写Java HashMap核心源码 上一章手写LinkedList核心源码,本章我们来手写Java HashMap的核心源码. 我们来先了解一下HashMap的原理.HashMap 字面意思 has ...
- java面试:手写代码
二分查找法. /** * 二分查找法:给定一组有序的数组,每次都从一半中查找.直到找到要求的数据. * 主要是得找到下标的表示方法. */ public class BinaryFind { /** ...
- java面试之手写单例模式
为什么要有单例模式 实际编程应用场景中,有一些对象其实我们只需要一个,比如线程池对象.缓存.系统全局配置对象等.这样可以就保证一个在全局使用的类不被频繁地创建与销毁,节省系统资源. 实现单例模式的几个 ...
- 2 手写Java LinkedList核心源码
上一章我们手写了ArrayList的核心源码,ArrayList底层是用了一个数组来保存数据,数组保存数据的优点就是查找效率高,但是删除效率特别低,最坏的情况下需要移动所有的元素.在查找需求比较重要的 ...
- 1 手写Java ArrayList核心源码
手写ArrayList核心源码 ArrayList是Java中常用的数据结构,不光有ArrayList,还有LinkedList,HashMap,LinkedHashMap,HashSet,Queue ...
随机推荐
- hdu-2027题&&gets/getchar的区别
hdu-2027题(水题~~~) 统计每个元音字母在字符串中出现的次数. Input输入数据首先包括一个整数n,表示测试实例的个数,然后是n行长度不超过100的字符串. Output对于每个测试实例输 ...
- Git:二、本地文件操作
文件必须放在本地Git仓库的文件夹下,子文件夹也可以. 1.添加/修改 git add <文件名> 2.提交 git commit -m "本次提交说明" 可以add很 ...
- 使用Visual Studio Team Services进行压力和性能测试(一)——创建基础的URL压力测试
使用Visual Studio Team Services进行压力和性能测试(一)--创建基础的URL压力测试 概述 压力测试使应用程序更加健壮,并审核在用户负载下的行为,这样我们可以在当前的基础设施 ...
- 手把手教你全家桶之React(三)--完结篇
前言 本篇主要是讲一些全家桶的优化与完善,基础功能上一篇已经讲得差不多了.直接开始: Source Maps 当javaScript抛出异常时,我们会很想知道它发生在哪个文件的哪一行.但是webpac ...
- 关于wordpress慢的问题
随笔记载: 关于wordpress访问慢的原因,主要是wordpress采用了google前端库, 需替换wordpress/wp-includes/script-header.php中所有的goog ...
- NVIDIA显卡笔记本安装ubuntu驱动以及分辨率之详解
随着对ubuntu的了解,突然想在自己的笔记本上装一个双系统.在网上查了安装方法之后,发现因为nvidia显卡的原因会出现一些问题,结果在我自己装了之后发现问题要比看到的多,再看了无数个帖子之后,最终 ...
- MySQL数据库有哪些安全相关的参数需要修改?
https://dev.mysql.com/doc/refman/5.7/en/security-options.htmlhttps://dev.mysql.com/doc/refman/5.7/en ...
- c/c++ 线性表之单向链表
c/c++ 线性表之单向链表 线性表之单向链表 不是存放在连续的内存空间,链表中的每个节点的next都指向下一个节点,最后一个节点的下一个节点是NULL. 真实的第一个节点是头节点,头节点不存放数据, ...
- php 计算出一年中每周的周一日期
最近接到一个任务,归纳起来,就是:要算出每年当中,每周的周一日期.想了一会,看了下date函数,深入了解了一下date函数各个参数的含义之后,终于把这道题做出来了! 在date()函数中,有一个参数对 ...
- March 06th, 2018 Week 10th Tuesday
Hope for the best, but prepare for the worst. 抱最好的愿望,做最坏的打算. To hope for the best and prepare for th ...