Java浮点数计算精确度问题

2024-05-08 14:38

本文主要是介绍Java浮点数计算精确度问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在Java中如果使用浮点型小数进行计算,尤其是商业计算将会出现奇怪的结果。比如下面的程序:

System.out.println(30.29 - 1.01);

我们期望它输出的结果是29.28,但是实际结果确是:29.279999999999998。
首先解释下为什么,为什么整数的时候不会呢,只有浮点数float,double才会呢,而且有的时候会,有的时候不会?
比如下面的程序:

System.out.println(30.25 - 1.125);

我们期望它输出的结果是29.125,实际结果也是:29.125。

这是为什么呢?

原因是因为在二进制中无法精确的表示十进制的小数,比如我们可以精确的表示0.25,0.125。但是无法精确的表示0.2.这也解释了为什么有些浮点数相减不会出现精确度的问题,而有些会。其根本原因就是二进制无法精确的表示所有的十进制小数部分。

关于浮点数为什么不精确,可以参考这篇文章:
http://www.th7.cn/Program/java/201606/873809.shtml

还好,Java为我们提供了用于商业计算且能够保留精确度的BigDecimal.
但是一定要注意在将浮点数转化为BigDecimal时,传入的构造参数一定要是String类型的,否则计算结果也是会出现精确度问题

  double d1 = 30.29;double d2 = 1.01;System.out.println(new BigDecimal(d1));System.out.println(new BigDecimal(d2));System.out.println(new BigDecimal(d1).subtract(new BigDecimal(d2)));

上面代码的输出结果如下:
这里写图片描述

所以一定不要以为使用BigDecimal计算的结果就是精确的。出现上面的结果是因为在构造BigDecimal时传入的是double,这样构造出来的BigDecimal已经不是我们预期 结果了。

所以千万记得,浮点数计算时,构造BigDecimal时一定要传入String参数,而不能是浮点数。

  double d1 = 30.29;double d2 = 1.01;System.out.println(a3);System.out.println(a4);System.out.println(a3.subtract(a4));

上面的代码输出的结果,与预期一致
这里写图片描述

另外BigDecimal还提供了加减乘除的运算,并且提供了四舍五入的方法。DecimalFormat 可以用于格式化数据。

public class TestBigDecimal {public static void main(String[] args) {//  Double.toString()  可以将double类型转化为StringSystem.out.println(new BigDecimal("0.06"));BigDecimal a1 = new BigDecimal("123.2562");BigDecimal a2 = new BigDecimal("0.75");System.out.println(a1.scale());//除System.out.println(a1.divide(a2, 2));//加System.out.println(a1.add(a2));//乘System.out.println(a1.multiply(a2));//减System.out.println(a1.subtract(a2));System.out.println(new BigDecimal("2.3").setScale(2));//只入不舍System.out.println("只入不舍" + a1.setScale(2, RoundingMode.UP));//只舍不入System.out.println("只舍不入" + a1.setScale(2, RoundingMode.DOWN));//四舍五入System.out.println("四舍五入" + a1.setScale(2, RoundingMode.HALF_UP));String number = "89" + "";DecimalFormat df2 = (DecimalFormat) DecimalFormat.getInstance();//强制保留两位小数,不足的补0df2.applyPattern("0.00");System.out.println(df2.format(Double.parseDouble(number)));double d1 = 30.29;double d2 = 1.01;System.out.println(new BigDecimal(d1));System.out.println(new BigDecimal(d2));System.out.println(new BigDecimal(d1).subtract(new BigDecimal(d2)));BigDecimal a3 = new BigDecimal(d1 + "");BigDecimal a4 = new BigDecimal(d2 + "");System.out.println(a3);System.out.println(a4);System.out.println(a3.subtract(a4));System.out.println(30.29 - 1.01);System.out.println(d1 - d2);}
}

这篇关于Java浮点数计算精确度问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/970604

相关文章

Nginx启动失败:端口80被占用问题的解决方案

《Nginx启动失败:端口80被占用问题的解决方案》在Linux服务器上部署Nginx时,可能会遇到Nginx启动失败的情况,尤其是错误提示bind()to0.0.0.0:80failed,这种问题通... 目录引言问题描述问题分析解决方案1. 检查占用端口 80 的进程使用 netstat 命令使用 ss

Java中对象的创建和销毁过程详析

《Java中对象的创建和销毁过程详析》:本文主要介绍Java中对象的创建和销毁过程,对象的创建过程包括类加载检查、内存分配、初始化零值内存、设置对象头和执行init方法,对象的销毁过程由垃圾回收机... 目录前言对象的创建过程1. 类加载检查2China编程. 分配内存3. 初始化零值4. 设置对象头5. 执行

SpringBoot整合easy-es的详细过程

《SpringBoot整合easy-es的详细过程》本文介绍了EasyES,一个基于Elasticsearch的ORM框架,旨在简化开发流程并提高效率,EasyES支持SpringBoot框架,并提供... 目录一、easy-es简介二、实现基于Spring Boot框架的应用程序代码1.添加相关依赖2.添

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程

《SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程》本文详细介绍了如何在虚拟机和宝塔面板中安装RabbitMQ,并使用Java代码实现消息的发送和接收,通过异步通讯,可以优化... 目录一、RabbitMQ安装二、启动RabbitMQ三、javascript编写Java代码1、引入

spring-boot-starter-thymeleaf加载外部html文件方式

《spring-boot-starter-thymeleaf加载外部html文件方式》本文介绍了在SpringMVC中使用Thymeleaf模板引擎加载外部HTML文件的方法,以及在SpringBoo... 目录1.Thymeleaf介绍2.springboot使用thymeleaf2.1.引入spring

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui