【Java开发常见错误】数值计算精度和舍入问题
admin
2024-01-31 04:31:50
0

今天单独分享数值计算的问题,是因为最近处理一次线上服务告警时,发现还有很多同学不了解浮点数计算的坑。

数值精度问题引发的Bug一般难以发现,所以我们在公司处理这方面的业务时一定要特别注意。

下面我们来具体看看这些问题。

数值精度问题

下面输出的结果是 ture 还是 false ?

 public static void main(String[] args) {Double num1 = 0.15;Double num2 = 0.05;System.out.println(num1 % num2 == 0);}

正确答案是 false 。这是因为计算机无法精确的保存浮点数,所以浮点数计算的结果也不可能精准。

再来看一段代码猜猜输出结果。

public static void main(String[] args) {System.out.println(0.1+0.2);System.out.println(1.0-0.8);System.out.println(4.015*100);System.out.println(123.3/100);
}

输出结果如下:

0.30000000000000004
0.19999999999999996
401.49999999999994
1.2329999999999999

输出结果和我们预期的很不一样,出现这种问题的原因是因为计算机是以二进制存储数值的,浮点数也不例外。Java采用了IEEE754标准实现浮点数的表达和运算。

比如,0.1 的二进制表示为 0.0 0011 0011 0011… (0011 无限循环),再转换为十进制就是 0.1000000000000000055511151231257827021181583404541015625。对于计算机而言,0.1 无法精确表达,这是浮点数计算造成精度损失的根源。

你可能会觉得,这种相差非常小不会对产生多大影响,但如果把损失的精度换算成金钱,每天有上百万交易,每次交易都差一分钱,一个月下来就是30万。

数值舍入问题

下面这段代码的输出结果是什么?

public static void main(String[] args) {double num = 3.35;System.out.println(String.format("%.1f", num));
}

输出结果

3.4

这就是由精度问题和舍入方式共同导致的,double 3.35 其实相当于 3.350xxx

String.format 采用四舍五入的方式进行舍入,取 1 位小数,double 的 3.350 四舍五入为 3.4

解决方案

涉及到浮点数精确表达和运算的场景,使用BigDecimal类型。

但是注意在使用BigDecimal的时候也有几个坑要避开。

第一个坑:

使用 BigDecimal 表示和计算浮点数,务必使用字符串的构造方法来初始化 BigDecimal。

  public static void main(String[] args) {System.out.println(new BigDecimal("0.1").add(new BigDecimal("0.2")));System.out.println(new BigDecimal(0.1).add(new BigDecimal(0.2)));}

输出结果

0.3
0.3000000000000000166533453693773481063544750213623046875

第二个坑:

浮点数的字符串格式化也要通过 BigDecimal 进行。

   public static void main(String[] args) {double num = 3.35;System.out.println(String.format("%.1f", num));BigDecimal num1 = new BigDecimal("3.35");BigDecimal num2 = num1.setScale(1, BigDecimal.ROUND_DOWN);System.out.println(num2);}

输出结果

3.4
3.3

总结

第一,要精确表示浮点数应该使用 BigDecimal。并且使用 String 入参的构造方法或者 BigDecimal.valueOf 方法来初始化。

第二,对浮点数做精确计算,参与计算的各种数值应该始终使用 BigDecimal,所有的计算都要通过 BigDecimal 的方法进行,任何一个环节出现精度损失,最后的计算结果可能都会出现误差。

第三,对于浮点数的格式化,建议使用 BigDecimal 来表示浮点数,并使用其 setScale 方法指定舍入的位数和方式。

作者简介

鑫茂,2022年3月参加工作,从事后端开发。

高度自律,中度代码洁癖,喜欢Java,看到美的东西就会拼命研究。闲暇之余,喜读思维方法、哲学心理学以及历史等方面的书,偶尔写些文字。

希望通过文章,结识更多同道中人。

参考资料

  • [2] BigDecimal 源码
  • [3]《数值计算》

相关内容

热门资讯

汉口银行储蓄国债(电子式)手机... 2026年4月,经财政部和中国人民银行批准,汉口银行在湖北地方法人银行中首家上线储蓄国债(电子式)手...
AI引爆存储狂潮,全球芯片牛股... AI引爆超级牛市,储存芯片彻底涨疯了。 隔夜,美股芯片巨头强劲财报,再次引爆华尔街热情。 周二美股盘...
半导体板块领涨,消费电子ETF... 截至5月6日10点13分,上证指数涨1.11%,深证成指涨2.39%,创业板指涨3.34%。国家大基...
原创 俄... 在阅读本文前,麻烦各位先点下关注,方便之后我们共同讨论与分享。作为作者,我会不负所托,按时产出更高质...
俞浩:在中国,只有雷军、余承东... 一个做扫地机器人的,竟然要去做汽车和火箭?你会觉得追觅太飘了,不够脚踏实地吗? 今日追觅创始人兼CE...
从“五一”假期触摸中国经济发展... 原标题:活力奔涌动能足——从“五一”假期触摸中国经济发展脉动 立夏之际,中国大地生机勃发,处处跃动...
贺博生:5.6黄金暴涨原油暴跌... 投资市场永远有四个层次:保住本金,控制风险,赚取收益,长期稳定持续赢利。不要因为一天的输赢定结果,赚...
《异环》首日流水过亿,完美世界... ?2025年,完美世界交出了一份亮眼的成绩单。根据2025年年报,报告期内公司实现营业收入66.60...
白癜风医生刘云涛:盛夏养护不松... 盛夏是白癜风养护的关键期,也是巩固复色成果的黄金期。很多患者经过一段时间的治疗,白斑出现了复色迹象,...
SpaceX上市催化,卫星ET... 截至5月6日10点18分,上证指数涨1.20%,深证成指涨2.49%,创业板指涨3.49%。国家大基...
“排队3小时,打卡1分钟”,多... 来源:澎湃新闻 5月5日,首个叠加“春假”的“加长版五一”假期收官。当日,多个景区披露的数据显示,假...
科创板走强,多只科创50相关E... 科创板走强,海光信息涨20%,澜起科技、佰维存储涨超16%,寒武纪涨超11%。 受盘面影响,多只科创...
原创 巴... 2026巴菲特股东大会正式结束,巴菲特在60年来首次坐在台下担任特殊观众。这一次,巴菲特并没有发表太...
港股创新药概念股走弱,相关ET... 港股创新药概念股走弱,三生制药跌超3%,信达生物、石药集团跌超2%。 受盘面影响,港股创新药相关ET...
坎宁安23分哈登22+8+7 ... 【搜狐体育战报】北京时间5月6日NBA季后赛,主场作战的活塞以111-101击败骑士。杜伦11分12...
原创 潘... 现如今,能成功预测房地产市场趋势的人并不多,潘石屹就是其中之一。前些年,潘石屹就提出楼市的“黄金时代...
年度行情促“百亿券商”扩容,中... 上市券商全部“交卷”,去年和今年一季度业绩情况也随之揭晓。 2025年A股市场整体震荡上行,股权融资...
严选FOF强势崛起 银行重塑财... 近期,公募 FOF 市场迎来爆发式增长,多家银行加速布局严选 FOF、定制 FOF 产品。依托完善的...
“AI烧钱大战”依旧如火如荼!... 通财经APP获悉,美股市场七大科技巨头(即Magnificent 7)中的除英伟达之外的六大巨头已经...