iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >详解java中BigDecimal精度问题
  • 737
分享到

详解java中BigDecimal精度问题

2024-04-02 19:04:59 737人浏览 泡泡鱼

Python 官方文档:入门教程 => 点击学习

摘要

目录一、背景二、BigDecimal构造函数1、四种构造函数2、为什么会出现这种情况3、如何解决三、常用方法1、常用方法2、取舍规则四、格式化一、背景 在实际开发中,对于 不需要任何

一、背景

在实际开发中,对于 不需要任何准确计算精度的属性可以直接使用float或double,但是如果需要精确计算结果,则必须使用BigDecimal,例如价格、质量。

为什么这么说,主要有两点

1、double计算会有精度丢失问题

2、在除法运算时,BigDecimal提供了丰富的取舍规则。(double虽然可以通过NumberFORMat进行四舍五入,但是NumberFormat是线程安全的)

对于精度问题我们可以看下实际的例子


public static void main(String[] args) {
    //正常 3.3
    System.out.println("加法结果:"+(1.1+2.2));
    //正常 -7.9
    System.out.println("减法结果:"+(2.2-10.1));
    //正常 2.42
    System.out.println("乘法结果:"+(1.1*2.2));
    //正常 0.44
    System.out.println("除法结果:"+(4.4/10));
}

实际控制台输出

为什么会这样

在于我们的计算机是二进制的。浮点数没有办法是用二进制进行精确表示。我们的CPU表示浮点数由两个部分组成:指数和尾数,这样的表示方法一般都会

失去一定的精确度,有些浮点数运算也会产生一定的误差。如:2.4的二进制表示并非就是精确的2.4。反而最为接近的二进制表示是 2.3999999999999999。

浮点数的值实际上是由一个特定的数学公式计算得到的。

二、BigDecimal构造函数

1、四种构造函数


BigDecimal(int)     //创建一个具有参数所指定整数值的对象。
BigDecimal(double)  //创建一个具有参数所指定双精度值的对象。
BigDecimal(long)    //创建一个具有参数所指定长整数值的对象。
BigDecimal(String)  //创建一个具有参数所指定以字符串表示的数值的对象。

这几个都是常用的构造器,他们返回的对象都是BigDecimal对象。换而言之,将BigDecimal对象转换为其他类型的对象,我们通过以下几种。


toString()          //将BigDecimal对象的数值转换成字符串。
doubleValue()       //将BigDecimal对象中的值以双精度数返回。
floatValue()        //将BigDecimal对象中的值以单精度数返回。
longValue()         //将BigDecimal对象中的值以长整数返回。
intValue()          //将BigDecimal对象中的值以整数返回。

这里需要非常注意BigDecimal(double)的构造函数,也是会存在精度丢失的问题,其它的不会,这里也可以举例说明


public static void main(String[] args) {
    BigDecimal intDecimal = new BigDecimal(10);
    BigDecimal doubleDecimal = new BigDecimal(4.3);
    BigDecimal longDecimal = new BigDecimal(10L);
    BigDecimal stringDecimal = new BigDecimal("4.3");
    System.out.println("intDecimal=" + intDecimal);
    System.out.println("doubleDecimal=" + doubleDecimal);
    System.out.println("longDecimal=" + longDecimal);
    System.out.println("stringDecimal=" + stringDecimal);
}

控制台实际输出

从图中很明显可以看出,对于double的构造函数是会存在精度丢失的可能的。

2、为什么会出现这种情况

这个在new BigDecimal(double)类型的构造函数上的注解有解释说明。

这个构造函数的结果可能有些不可预测。 可以假设在Java中写入new BigDecimal(0.1)创建一个BigDecimal ,它完全等于0.1(非标尺值为1,比例为1),但实际上等于

0.1000000000000000055511151231257827021181583404541015625。 这是因为0.1不能像double (或者作为任何有限长度的二进制分数)精确地表示。

因此,正在被传递给构造的值不是正好等于0.1。

3、如何解决

有两种常用的解决办法。

1、是将double 通过Double.toString(double)先转为String,然后放入BigDecimal的String构造函数中。

2、不通过BigDecimal的构造函数,而是通过它的静态方法BigDecimal.valueOf(double),也同样不会丢失精度。

示例


 public static void main(String[] args) {
    String string = Double.toString(4.3);
    BigDecimal stringBigDecimal = new BigDecimal(string);
    BigDecimal bigDecimal = BigDecimal.valueOf(4.3);
    System.out.println("stringBigDecimal = " + stringBigDecimal);
    System.out.println("bigDecimal = " + bigDecimal);
}

运行结果

这样也能保证,对与double而言,转BigDecimal不会出现精度丢失的情况。

三、常用方法

1、常用方法

示例


public static void main(String[] args) {
    BigDecimal a = new BigDecimal("4.5");
    BigDecimal b = new BigDecimal("1.5");
    BigDecimal c = new BigDecimal("-10.5");

    BigDecimal add_result = a.add(b);
    BigDecimal subtract_result = a.subtract(b);
    BigDecimal multiply_result = a.multiply(b);
    BigDecimal divide_result = a.divide(b);
    BigDecimal remainder_result = a.remainder(b);
    BigDecimal max_result = a.max(b);
    BigDecimal min_result = a.min(b);
    BigDecimal abs_result = c.abs();
    BigDecimal negate_result = a.negate();

    System.out.println("4.5+1.5=" + add_result);
    System.out.println("4.5-1.5=" + subtract_result);
    System.out.println("4.5*1.5=" + multiply_result);
    System.out.println("4.5/1.5=" + divide_result);
    System.out.println("4.5/1.5余数=" + remainder_result);
    System.out.println("4.5和1.5最大数=" + max_result);
    System.out.println("4.5和1.5最小数=" + min_result);
    System.out.println("-10.5的绝对值=" + abs_result);
    System.out.println("4.5的相反数=" + negate_result);
}

4.5+1.5=6.0

4.5-1.5=3.0

4.5*1.5=6.75

4.5/1.5=3

4.5/1.5余数=0.0

4.5和1.5最大数=4.5

4.5和1.5最小数=1.5

-10.5的绝对值=10.5

4.5的相反数=-4.5

这里把除法单独再讲一下,因为除法操作的时候会有除不尽的情况,,比如 3,5/3,这时会报错java.lang.ArithmeticException: Non-terminating decimal expansion;

no exact representable decimal result。所以这里要考虑除不尽的情况下,保留几位小数,取舍规则。(除法如果可能存在除不进,那就用下面方法)


BigDecimal divide(BigDecimal divisor, int scale, int roundingMode) 第一参数表示除数,第二个参数表示小数点后保留位数,第三个参数表示取舍规则。

2、取舍规则


ROUND_UP          //不管保留数字后面是大是小(0除外)都会进1
ROUND_DOWN        //保留设置数字,后面所有直接去除
ROUND_HALF_UP     //常用的四舍五入 
ROUND_HALF_DOWN   //五舍六入
ROUND_CEILING     //向正无穷方向舍入
ROUND_FLOOR       //向负无穷方向舍入
ROUND_HALF_EVEN   //向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,如果保留位数是奇数,使用ROUND_HALF_UP,如果是偶数,使用ROUND_HALF_DOWN
ROUND_UNNECESSARY //计算结果是精确的,不需要舍入模式 

注意 我们最常用的应该是 ROUND_HALF_UP(四舍五入)

这里举几个常用的取舍规则


 public static void main(String[] args) {

    BigDecimal a = new BigDecimal("1.15");
    BigDecimal b = new BigDecimal("1");

    //不管保留数字后面是大是小(0除外)都会进1 所以这里输出为1.2
    BigDecimal divide_1 = a.divide(b,1,BigDecimal.ROUND_UP);
    //保留设置数字,后面所有直接去除         所以这里输出为1.1
    BigDecimal divide_2 = a.divide(b,1,BigDecimal.ROUND_DOWN);
    //常用的四舍五入         所以这里输出1.2
    BigDecimal divide_3 = a.divide(b,1,BigDecimal.ROUND_HALF_UP);
    //这个可以理解成五舍六入   所以这里输出1.1
    BigDecimal divide_4 = a.divide(b,1,BigDecimal.ROUND_HALF_DOWN);
    //这里将1.15改成1.16
    BigDecimal c = new BigDecimal("1.16");
    //那么这里就符合六入了 所以输出变为1.2
    BigDecimal divide_5 = c.divide(b,1,BigDecimal.ROUND_HALF_DOWN);
    System.out.println("divide_1 = " + divide_1);
    System.out.println("divide_2 = " + divide_2);
    System.out.println("divide_3 = " + divide_3);
    System.out.println("divide_4 = " + divide_4);
    System.out.println("divide_5 = " + divide_5);

}

运行结果

divide_1 = 1.2

divide_2 = 1.1

divide_3 = 1.2

divide_4 = 1.1

divide_5 = 1.2

四、格式化

由于NumberFormat类的format()方法可以使用BigDecimal对象作为其参数,可以利用BigDecimal对超出16位有效数字的货币值,百分值,以及一般数值进行格式化控制。

以利用BigDecimal对货币和百分比格式化为例。首先,创建BigDecimal对象,进行BigDecimal的算术运算后,分别建立对货币和百分比格式化的引用,最后利用

BigDecimal对象作为format()方法的参数,输出其格式化的货币值和百分比。

示例


public static void main(String[] args) {
    //建立货币格式化引用
    NumberFormat currency = NumberFormat.getCurrencyInstance();
    //建立百分比格式化引用
    NumberFormat percent = NumberFormat.getPercentInstance();
    //百分比小数点最多3位
    percent.setMaximumFractionDigits(3);
    //取整
    NumberFormat integerInstance = NumberFormat.getIntegerInstance();
    ////金额
    BigDecimal loanAmount = new BigDecimal("188.555");
    ////利率
    BigDecimal interestRate = new BigDecimal("0.018555555");
    //没有指定保留位数的情况下 默认保留2位
    System.out.println("金额: " + currency.format(loanAmount));
    //货币(百分比)格式化   指定默认的取舍规则是四舍五入
    System.out.println("利率: " + percent.format(interestRate));
    //取整还有点不一样 188.555取整为189, 188.51也是189 但是189.5确实188,所以它不是真正意义上的四舍五入
    System.out.println("取整: " + integerInstance.format(loanAmount));
}

运行结果

金额: ¥188.56利率: 1.856%取整: 189

这里有几点在说明下

1、格式化的时候没有指定保留位数的情况下 默认保留2位。

2、货币(百分比)格式化 指定默认的取舍规则是四舍五入。

3、取整还有点不一样 188.555取整为189, 188.51也是189 但是189.5确实188,所以它不是真正意义上的四舍五入。

以上就是详解java中BigDecimal精度问题的详细内容,更多关于java的资料请关注编程网其它相关文章!

--结束END--

本文标题: 详解java中BigDecimal精度问题

本文链接: https://www.lsjlt.com/news/125641.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 详解java中BigDecimal精度问题
    目录一、背景二、BigDecimal构造函数1、四种构造函数2、为什么会出现这种情况3、如何解决三、常用方法1、常用方法2、取舍规则四、格式化一、背景 在实际开发中,对于 不需要任何...
    99+
    2024-04-02
  • java中的BigDecimal精度问题怎么解决
    在Java中,可以使用BigDecimal类来解决精度问题。BigDecimal类提供了高精度的数值计算,可以避免浮点数计算精度丢失...
    99+
    2023-08-16
    java BigDecimal
  • java中BigDecimal精度问题的示例分析
    这篇文章给大家分享的是有关java中BigDecimal精度问题的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、背景在实际开发中,对于 不需要任何准确计算精度的属性可以直接使用float或double...
    99+
    2023-06-15
  • Java中BigDecimal除法使用不当导致精度问题
    目录I. 问题抛出 II. 源码定位 1. 整形传参构造 2. 浮点传参 3. String传参 4. 小结在使用BigDecimal的除法时,遇到一个鬼畜的问题,本以为的精度计算,...
    99+
    2024-04-02
  • Java精确计算BigDecimal类详解
    引言 float和double类型的主要设计目标是为了科学计算和工程计算。他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提供完...
    99+
    2024-04-02
  • SpringBoot解决BigDecimal传到前端后精度丢失问题
    目录简介问题描述实例问题复现Java后端BigDecimal的范围解决方案方案1:全局处理方案2:局部处理简介 本文用示例介绍SpringBoot如何解决BigDecimal传到前端...
    99+
    2024-04-02
  • 详解Python中的数据精度问题
    目录一、python运算时精度问题1.运行时精度问题2.解决方案:添加方法二、python四舍五入时精度问题1.使用round与浮点数格式化时候的精度问题 2.解决方案,使...
    99+
    2024-04-02
  • java精度问题怎么解决
    在Java中,处理浮点数的精度问题可以使用BigDecimal类来解决。BigDecimal类提供了精确的数值计算,可以避免浮点数的...
    99+
    2023-08-16
    java
  • SpringBoot如何解决BigDecimal传到前端后精度丢失问题
    这篇文章主要介绍“SpringBoot如何解决BigDecimal传到前端后精度丢失问题”,在日常操作中,相信很多人在SpringBoot如何解决BigDecimal传到前端后精度丢失问题问题上存在疑惑,小编查阅了各式资料,整理出简单好用的...
    99+
    2023-07-01
  • Java中求Logn/log2 的精度问题
    目录java求Logn/log2精度例如程序如下:java 处理高精度计算使用的BigDecimal类的时候需要注意的地方:java求Logn/log2精度 经过本人测试,java ...
    99+
    2024-04-02
  • java中计算缺失精度的问题
    Java中计算会缺失精度的主要原因是浮点数的精度问题和整数溢出问题。 浮点数的精度 Java中的浮点数类型为float和double,它们采用的是IEEE 754规范的浮点数编码,这种编码方式虽然能够表示大范围的实数,但存在一定的精度损失。...
    99+
    2023-09-16
    java jvm 开发语言
  • java中BigDecimal用法详解
    首先,学习一个东西,我们都必须要带着问题去学,这边我分为 【为什么?】【是什么?】【怎么用?】 【为什么要用BigDecimal?】 首先,我们先看一下,下面这个现象 那为什么会出...
    99+
    2024-04-02
  • Java高精度计算BigDecimal类实例分析
    这篇文章主要介绍了Java高精度计算BigDecimal类实例分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java高精度计算BigDecimal类实例分析文章都会有所收获,下面我们一起来看看吧。如果我们编...
    99+
    2023-06-03
  • java中BigDecimal的介绍及使用,BigDecimal格式化,BigDecimal常见问题
    文章目录 一、BigDecimal概述二、BigDecimal常用构造方法三、BigDecimal常用方法四、BigDecimal进阶BigDecimal的八种舍入模式BigDecimal格式化、小数点转换货币格式化与百分比格式化 ...
    99+
    2023-08-18
    java BigDecimal BigDecimal的使用及坑
  • 一个Java中BigDecimal的问题记录
    题主今天在用 BigDecimal 进行计算的时候,遇到一个问题,那就是不管怎么计算,最后得到的值都没有变化,这里题主就有些疑惑了,用的也是推荐的API,先看代码: BigDec...
    99+
    2024-04-02
  • Java中的高精度计算怎么利用BigDecimal实现
    本篇文章给大家分享的是有关Java中的高精度计算怎么利用BigDecimal实现,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。首先看如下代码示例:System.out.prin...
    99+
    2023-05-31
    java ava bigdecimal
  • JS中toFixed()方法四舍五入的精度问题详解
    目录踩的坑 填坑方法 什么样的坑? 总结踩的坑 最近工作中,在计算一个商品的折扣价格,有时候总是出现价格会有一分钱的差异,涉及钱的问题都是比较敏感的,经过排查,最后发现竟然是 JS...
    99+
    2024-04-02
  • Java BigDecimal案例详解
    引言   float和double类型的主要设计目标是为了科学计算和工程计算。他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提...
    99+
    2024-04-02
  • 解决java.sql.Timestamp丢失精度的问题
    目录java.sql.Timestamp丢失精度需要重新补偿java.sql.Timestamp类的使用1. 获取当前时间2.String类型转换为Timestamp3.Timest...
    99+
    2024-04-02
  • springboot Long 精度丢失问题解决
    目录前言解决方法基于注解@JsonSerialize(不推荐)基于jackson全局配置(不推荐)使用JsonComponent 序列化配置WebMvcConfigurationSu...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作