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

前言

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

​ 听到这个问题,脑子里竟然突然有些懵,回答道:double用着顺手所以就用了,面试过后我自己在听录音复盘时(远程线上面试)听到自己的回答不由得扶额苦笑,后面又对这一块的内容进行了回顾加深。

double和float的区别

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

​ 1)有效数字位数不同

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

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

​ 也就是说因为有效数字位数不同,所以双精度的double要比单精度的float要更精准一些。

​ 2)数值取值范围不同

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

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

​ 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这种数据类型来表示金额。

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

总结

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

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

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: Spring通过依赖注入的方式来解决循环依赖题,在创建bean的同时进行属性的注入,从而避免了循环依赖的情况。当Spring在创建bean的过程中出现循环依赖时,它会将尚未创建完成的bean对象提前暴露给Spring容器,Spring容器在完成对所有bean的创建后,会利用这些提前暴露的bean对象来完成循环依赖的注入。 ### 回答2: 在面试过程中,当面试官到如何解决Spring中的循环依赖题,有效的回答应该包括以下几点。 首先,要明确循环依赖是指两个或多个Bean之间互相依赖,形成一个循环链,无法确定依赖关系的题。Spring为了解决这个题,采用了三级缓存(bean map)的方式。 其次,可以提到Spring的处理循环依赖是通过"提前暴露半成品实例"的方式来实现的。Spring容器在初始化装配Bean的过程中,首先创建Bean的实例,但不进行属性填充,在这个过程中,Spring会将刚刚创建的Bean实例进行缓存。然后,Spring继续对其他Bean进行创建,当创建到有对其他Bean的依赖注入时,Spring会检测到循环依赖的情况,然后从缓存中提取半成品实例,即已经存在的Bean实例,完成对依赖的注入,最终完成装配。 再次,可以提及Spring在解决循环依赖时使用了"构造器注入"和"属性注入"的方式。在存在循环依赖的情况下,采用构造器注入可以在Bean创建时就解决循环依赖题,而属性注入则需要在Bean的创建之后再进行依赖注入。 最后,需要指出Spring的循环依赖处理能力是有限的,并不适用于所有情况。如果循环依赖的链过长或存在复杂的依赖关系,容易导致死锁或循环等待的题。因此,在设计应用程序时,尽量避免循环依赖的产生,减少耦合度,提高系统的可维护性和扩展性。 综上所述,当面试官到Spring如何解决循环依赖题时,回答者可以通过解释Spring采用的三级缓存和"提前暴露半成品实例"的方式、使用构造器注入和属性注入的方式以及指出其局限性来回答这个题。 ### 回答3: 面试的过程中,如果面试官到Spring如何解决循环依赖的题,我可以这样回答: 首先,循环依赖是指两个或多个Bean之间相互依赖,形成了一个环状的依赖关系。例如,Bean A依赖于Bean B,而Bean B又依赖于Bean A。 在Spring中,解决循环依赖通常有两种方式。 首先是通过构造器注入来解决循环依赖。构造器注入是指在Bean实例化时通过构造器完成依赖注入。在这种情况下,Spring会检测到循环依赖并抛出一个异常,阻止Bean的创建。但是,如果我想要解决这个题,可以使用@Lazy注解来延迟Bean的实例化。通过延迟实例化,Spring能够在构建对象时避免产生循环依赖。 另一种方式是使用Setter方法注入来解决循环依赖。Setter方法注入是指在Bean实例化后,通过Setter方法完成依赖注入。在这种情况下,Spring允许先创建一个空的Bean对象,再通过使用Setter方法设置依赖关系。这种方式下,Spring能够在构建对象的时候完成所有的依赖注入。 总结来说,Spring通过使用构造器注入和Setter方法注入的方式来解决循环依赖的题。无论是采用哪种方式,我们都需要注意避免出现过多复杂的循环依赖关系,因为这可能会导致代码难以维护和理解。同时,需要确保Bean的依赖关系在整个应用程序中是清晰可见的,以便更好地管理和调试。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值