面试官问我存储金额应该用哪一种数据类型,我竟这样回答

前言

​ 最近在面试时,碰到这样一个问题:在问到项目部分时,面试官问我:你的项目中用到的分数、金额之类的数字是用的什么数据类型? 我没有过多思考脱口而出:double!随后面试官又问:为啥不用float?java

​ 听到这个问题,脑子里居然忽然有些懵,回答道:double用着顺手因此就用了,面试事后我本身在听录音复盘时(远程线上面试)听到本身的回答不禁得扶额苦笑,后面又对这一块的内容进行了回顾加深。web

double和float的区别

float(单精度浮点数)和double(双精度浮点数)的主要区别以下:面试

​ 1)有效数字位数不一样svg

​ 单精度浮点数有效数字为8位post

​ 双精度浮点数有效数字为16位spa

​ 也就是说由于有效数字位数不一样,因此双精度的double要比单精度的float要更精准一些。3d

​ 2)数值取值范围不一样code

​ 单精度浮点数的表示范围:-3.40E+38~3.40E+38xml

​ 双精度浮点数的表示范围:-1.79E+308~1.79E+308token

​ 3.40E+38的意思是3.4*10的38次方,而1.79E+308指的是1.79*10的308次方,因此double的取值范围要远远大于float

​ 3)内存中占有的字节数不一样

​ 单精度浮点数在内存中占4个字节

​ 双精度浮点数在内存中占8个字节

​ 也就是说双精度的double要比单精度的float更占内存

​ 4)在程序中的处理速度不一样

​ 通常来讲,CPU处理单精度浮点数的速度比处理双精度浮点数快

在程序中默认小数为double类型,因此若是要用float的话,必须进行强转

public static void mian(String[] args){
	float a = 1.1;	
}

好比我写了上面的代码的话,在程序中就会编译报错,正确的写法应该为以下的代码:

public static void mian(String[] args){
    float a = (float)1.1;
    float b = 1.1f;
}

手动强转或者在小数后面加f表示为float类型(f不区分大小写)

在使用float时须要注意一点:float 是8位有效数字,好比说有以下代码:

public static void main(String[] args){
    float a = 1.11111111111f;
    System.out.println(a);
}

最终的输出结果为:1.1111112

这里有一个疑问,不管第九位是否大于5,在取值的时候都会向第八位进1。

以上就是double和float的区别

金额到底应该用哪一种数据类型?

​ 在总结double和float的区别时,我发如今真实开发中针对金额的存储并不是如我以前思考的同样使用double或者float,为啥?让咱们看下面一段代码:

public static void main(String[] args) {
     double a=0.03;
     double b=0.02;
     double c=a-b;
     System.out.println(c);
}

​ 对于这段代码的执行结果,大部分人可能会想确定是0.01啊!可是运行以后会惊奇的发现结果竟然是0.009999999999999998,由于float与double都是浮点数,浮点数参与的运算一般伴随着由于没法精确表示而进行的近似与舍入,因此致使结果会有一些误差,而涉及到金额的计算是绝对不容许存在误差的。

​ 那么应该怎么表示金额呢?

​ 有两种解决方案:第一种是存储金额时以分或厘为单位存储一个整数,第二种是使用BigDecimal这种数据类型来表示金额。

​ 对于第一种是我目前在写项目时采用的,第二种暂时并未作尝试。

总结

面试官问的小小的一个问题居然藏有这么多玄机和学问,不禁得让我汗颜,归根结底仍是本身的知识面不够广。不过这也算是面试中的一些小小收获吧,能发现本身的不足并及时补足。

本文已在掘金同步上传:面试官问我存储金额应该用哪一种数据类型,我竟这样回答