【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]《数值计算》

相关内容

热门资讯

悄然历史新高!两市唯一的中证红... 1月20日,红利资产再度回暖。数据显示,截至14时6分,中证红利质量ETF(159209)涨0.64...
净流出,超400亿元! 【导读】1月19日股票ETF资金净流出超400亿元 中国基金报记者 若晖 股票ETF继续发挥稳定市场...
黄金vs.“数字黄金”:特朗普... 编者按 2024年比特币现货ETF获批拓宽了其投资渠道,特朗普连任后提出的比特币储备提案更推动其价格...
原创 中... 1月18日,日本共同社发布了一则引人注目的消息:中国正式收紧了对日本稀土出口的审查,并要求日本方面提...
最新消息!特朗普:将征收200... 据路透社1月20日报道,美国总统特朗普称将对法国葡萄酒和香槟征收200%的关税。 特朗普19日表示,...
淡马锡要走、屈臣氏要上:一宗迟... 日前,《华尔街日报》等外媒援引知情人士消息称,李嘉诚旗下长和(00001.HK)控股的屈臣氏集团正筹...
北京隅海岄价格坚挺,张晓龙选对... 北京进深 徐迪 2026年第三周(1.14-1.20),北京隅海岄完成4套网签,成交面积577.33...
财政部谈取消光伏等产品出口退税... 近期,财政部会同税务总局发布了公告,明确了自2026年4月1日起,取消光伏、磷化工等产品的出口退税,...
中国平煤神马控股集团增资至37... 天眼查工商信息显示,近日,中国平煤神马控股集团有限公司发生工商变更,注册资本由约194.3亿人民币增...
黄金年涨七成,银行保管箱从冷门... 【文/羽扇观金工作室】 2026年以来,全球地缘政治风险持续升温,黄金价格涨势不减,一个意想不到的...
“一人公司”调查 来源:中国新闻周刊 秦文山没想到,自己成了苏州AI创业领域第一个“吃螃蟹”的人。 2025年12月2...
【A股收评】大盘疲软,黄金股走... 1月20日,三大指数震荡走弱,截至收盘,上证指数跌0.01%,深证成指跌0.97%,创业板指跌1.7...
沪深300ETF打压大盘探新低... 沪深300ETF打压大盘探新低,震荡洗盘多头市场形态依然保持 今日消息面: 【发改委:将研究制定出台...
国家发改委:正在研究制定城乡居... “消费是就业和收入的函数,目前有关方面正在研究制定稳岗扩容提质行动和城乡居民增收计划,目的就是增强居...
开年最大核聚变融资诞生 | 上... 01 融资综述 加冕研究院据张通社Link数据库统计,1月12日–1月18日,上海企业共发生24起融...
打造传统与新兴产业协同增长格局... 世界银行发布的《全球经济展望》报告,为研判全球经济态势提供重要参考,整体传递出谨慎乐观的信号。报告预...
他是“80后任正非”,带出和华... 当人们将目光投向天空,看到那些优雅盘旋的飞行体时,脑海中最先浮现的,大概率会是那个银灰与白色交织的“...
原创 中... 1月16日,俄罗斯《生意人报》曝出了一条令人难以置信的消息,进入2026年后,中俄长达14年的电力进...
鸣鸣很忙启动招股 引入腾讯、淡... 上证报中国证券网讯(记者 夏子航)1月20日,我国最大的休闲食品饮料连锁零售商——湖南鸣鸣很忙商业连...
微银订购APP现货订购变非法期...   微银订购APP不正规也不合法!虚假宣传、诱导开户!而且他们的商品订购实际上是撮合式的双边期货交易...