点击上方“方志朋”,选择“设为星标”

回复”666“获取新整理的面试文章

本文来源:

juejin.im/post/5c08db5ff265da611e4d7417

《Java工程师面试突击(第3季)》重磅升级,由原来的70讲增至160讲,内容扩充一倍多,升级部分内容请参见文末

公司最近在做交易系统,交易系统肯定是要和钱打交道的,和钱有关,自然而然很容易想到用float存储,但是使用float存储金额做的计算是近似计算。

老板说:「用float做计算造成公司损失的钱都往你工资里扣。」

哼,扣工资就扣工资。但还是得静下心来想想为什么不能用float。

为什么不能使用float存储金额

首先看个例子:FloatTest.java

public class FloatTest {public static void main(String[] args) {float f1 = 6.6f;float f2 = 1.3f;System.out.println(f1 + f2);}
}

结果:7.8999996 和自己口算的值竟然不一样

计算机只认识0和1,所有类型的计算首先会转化为二进制的计算。

从计算机二进制角度计算 6.6 + 1.3 的过程

float底层存储

计算是由CPU来完成的,CPU表示浮点数由三部分组成 分为三个部分,符号位(sign),指数部分(exponent)和有效部分(fraction, mantissa)。其中float总共占用32位,符号位,指数部分,有效部分各占1位,8位,23位。

二进制的转化

对于实数,转化为二进制分为两部分,第一部分整数部分,第二部分是小数部分。整数部分计算二进制大家都很熟悉。

整数部分的计算:6转化为二进制

所以6最终的二进制为110

小数部分的计算

将小数乘以2,取整数部分作为二进制的值,然后再将小数乘以2,再取整数部分,以此往复循环。

0.6转化为二进制

…进入循环,循环体为1001 所以0.6转化为二进制为0.10011001… 6.6转化为二进制为110.10011001…

规约化

通过规约化将小数转为规约形式,类似科学计数法,就是保证小数点前面有一个有效数字。在二进制里面,就是保证整数位是一个1。110.10011001规约化为:1.1010011001*2^2。

指数偏移值

指数偏移值 = 固定值 + 规约化的指数值 固定值=2^(e-1)-1,其中的e为存储指数部分的比特位数,前面提到的float为8位。所以float中规定化值为127 6.6的二进制值规约化以后为1.1010011001*2^2,指数是2,所以偏移值就是127+2=129,转换为二进制就是10000001。

拼接6.6

6.6为正数,符号位为0,指数部分为偏移值的二进制10000001,有效部分为规约形式的小数部分,取小数的前23位即10100110011001100110011,最后拼接到一起即 01000000110100110011001100110011。

到这里已经大致可以知道float为什么不精确了,首先在存储的时候就会造成精度损失了,在这里小数部分的二进制是循环的,但是仍然只能取前23位。double造成精度损失的原因也是如此。推荐阅读:金融系统中正确的金额计算及存储方式。

求和

原来如此

不能使用float那用什么类型存储金额?

使用int 数据库存储的是金额的分值,显示的时候在转化为元。Java中的运算神器BigDecimal,这篇也推荐看下。

使用decimal mysql中decimal存储类型的使用

column_name  decimal(P,D);

D:代表小数点后的位数 P:有效数字数的精度,小数点也算一位 测试例子 数据表的创建:

CREATE TABLE `test_decimal` (`id` int(11) NOT NULL,`amount` decimal(10,2) NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4

对应的DAO层代码:TestDecimalDao.java

/*** @description dao层** @author JoyHe* @date 2018/11/05* @version 1.0*/
@Repository
public interface TestDecimalDao {@Select("select * from test_decimal where id = #{id}")TestDecimal getTestDecimal(int id);
}

测试类:TestDecimalDaoTest.java

/*** @description 测试类** @author JoyHe* @date 2018/11/05* @version 1.0*/
public class TestDecimalDaoTest extends BaseTest {@Resource  private TestDecimalDao testDecimalDao;@Test  public void test() {TestDecimal testDecimal1 =   testDecimalDao.getTestDecimal(1);TestDecimal testDecimal2 =   testDecimalDao.getTestDecimal(2);BigDecimal result =   testDecimal1.getAmount().add(testDecimal2.getAmount());System.out.println(result.floatValue());}
}

说明:jdbcType为decimal转化为javaType为BigDecimal 测试结果:

是符合预期的7.9

使用decimal存储类型的缺点

1、占用存储空间。

浮点类型在存储同样范围的值时,通常比decimal使用更少的空间

2、使用decimal计算效率不高  

因为使用decimal时间和空间开销较大,选用int作为数据库存储格式比较合适,可以同时避免浮点存储计算的不精确和decimal的缺点。对于存储数值较大或者保留小数较多的数字,数据库存储结构可以选择bigint。

热门内容:太强了!这款轻量级的数据库中间件完美解决了SpringBoot中分库分表问题
Spring Boot 还能“内存泄露”?排它!
学会 IDEA REST Client后,postman就可以丢掉了...
为什么老外不愿意用 MyBatis?最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡

用 float 存储金额,老板说损失从工资里扣!相关推荐

  1. mybatis float 小数0 不显示_卧槽!用 float 存储金额,老板说损失从工资里扣!

    开发者(KaiFaX) 我们都是开发者专注于前端.后端.大数据.区块链.人工智能的知识社区 来源:https://juejin.im/post/5c08db5ff265da611e4d7417 作者: ...

  2. 老板,用float存储金额为什么要扣我工资

    点击上方 好好学java ,选择 星标 公众号重磅资讯,干货,第一时间送达今日推荐:分享一套基于SpringBoot和Vue的企业级中后台开源项目,这个项目有点哇塞!个人原创100W +访问量博客:点 ...

  3. 不能用float、double 存储金额——BigDecimal详解

    1.为什么不能使用 float 存储金额? public class FloatTest {public static void main(String[] args) {float f1 = 6.6 ...

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

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

  5. Java存储金额解决方案BigDecimal

    使用BigDecimal来存储金额数据,数据库中使用decimal类型,长度18,小数点2. 在JPA中创建时如下: @Column(columnDefinition="decimal(18 ...

  6. 聪明的老板才不招工资低的程序员

    前阵子写了一篇<如果两个程序员差不多,选写作能力更好的那个>,就有读者留言说:"老板,不都是选工资更低的那个么?".其实,这是另一个维度上的看法,正好最近也和一些经常招 ...

  7. flask中的CBV , flask-session在redis中存储session , WTForms数据验证 , 偏函数 , 对象里的一些小知识...

    flask中的CBV , flask-session在redis中存储session , WTForms数据验证 , 偏函数 , 对象里的一些小知识 flask中的CBV写法 后端代码 # 导入vie ...

  8. 5 Android数据存储 任务二 应用程序数据文件夹里的文件读写 ,

    Android中提供了两个方法用来打开应用程序的数据文件夹IO流. 1.FileInputStream openFileInput(String name):参数name表示某个文件名,该方法用于打开 ...

  9. java金额类型_Java中存储金额用什么数据类型?

    很早之前, 记得一次面试, 面试官问存储金钱用什么数据类型? 当时只知道8种数据类型(boolean, byte, short, int, long, float, double, char)的我, ...

最新文章

  1. 有关指针的数据类型的小结
  2. java算法之冒泡排序法
  3. 再说javascript 的__proto__ 和prototype 属性
  4. redis ubuntu php 5.2,ubuntu 14.04下简易安装php5.5 + apache2 + redis + mysql
  5. NewCode----句子反转
  6. hybird之web动态换肤实现
  7. 指定的颜色信息显示方法
  8. WPF:window设置单一开启
  9. RD与RT MPLS
  10. python 实现A星算法
  11. Halcon常用图像预处理算子总结
  12. HTMl--基础样式的使用
  13. 解决GD32F105休眠后无法唤醒的问题
  14. Mac电脑怎么远程桌面连接?
  15. 关于SSD写放大问题
  16. Ubuntu18.04人工智能环境搭建
  17. 网页收藏栏小图标_如何设置在网页地址栏中的小图标
  18. 常用计算机字长,计算机基本单位——位、字节、字、字长
  19. 艰难的选择_如何艰难地保护Kubernetes
  20. 在css样式中隐藏元素,用JS改变的元素CSS样式,css里display :none 隐藏 block 显示

热门文章

  1. Mysql与Oracle区别
  2. CentOS VMware 配置IP小结 静态 配置 桥接 NAT
  3. [cocos2dx UI] CCLabelAtlas 为什么不显示最后一个字
  4. POJ - 3538 - Domestic Networks
  5. DB2 9 利用开辟(733 测验)认证指南,第 1 部分: 数据库工具与编程步调(6)
  6. C#和JavaScript的简单互交
  7. GPT-3再进化:通过浏览网页来提高事实准确率
  8. 11位科幻作家参与,首次AI人机共创写作实验启动
  9. 信息保留的二值神经网络IR-Net,落地性能和实用性俱佳 | CVPR 2020
  10. 抗击疫情!阿里云为加速新药疫苗研发提供免费AI算力