广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Java BigDecimal正确用法详解
  • 128
分享到

Java BigDecimal正确用法详解

2024-04-02 19:04:59 128人浏览 独家记忆

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

摘要

目录一、背景二、事故案例1、问题2、问题复现3、源码分析4、原因分析三、总结四、工具类一、背景 BigDecimal 平时主要用于计算金钱时,其自身提供了很多的构造方法,但是这些构造

一、背景

BigDecimal 平时主要用于计算金钱时,其自身提供了很多的构造方法,但是这些构造方法使用不当会造成精度丢失,从而引起事故。

二、事故案例

1、问题

收银台计算商品价格报错,导致订单无法支付

2、问题复现

public static void main(String[] args) {
    BigDecimal bigDecimal=new BigDecimal(88);
    System.out.println(bigDecimal);
    bigDecimal=new BigDecimal("8.8");
    System.out.println(bigDecimal);
    bigDecimal=new BigDecimal(8.8);
    System.out.println(bigDecimal);
}

3、源码分析

public static long doubleToLongBits(double value) {
    long result = doubleToRawLongBits(value);
    // Check for NaN based on values of bit fields, maximum
    // exponent and nonzero significand.
    if ( ((result & DoubleConsts.EXP_BIT_MASK) ==
          DoubleConsts.EXP_BIT_MASK) &&
         (result & DoubleConsts.SIGNIF_BIT_MASK) != 0L)
        result = 0x7ff8000000000000L;
    return result;
}

问题就处在 doubleToRawLongBits 这个方法上,在 jdk 中 double 类(float 与 int 对应)中提供了 double 与 long 转换,doubleToRawLongBits 就是将 double 转换为 long,这个方法是原始方法(底层不是 java 实现,是 c++ 实现的)。

4、原因分析

在 java 中 BigDecimal 处理数据时把十进制小数扩大 N 倍让它在整数上进行计算,并保留相应的精度信息。

  1. float 和 double 类型,主要是为了科学计算和工程计算而设计的,之所以执行二进制浮点运算,是为了在广泛的数值范围上提供较为精确的快速近和计算。
  2. 并没有提供完全精确的结果,所以不应该被用于精确的结果的场合。
  3. 当浮点数达到一定大的数,就会自动使用科学计数法,这样的表示只是近似真实数而不等于真实数。
  4. 当十进制小数位转换二进制的时候也会出现无限循环或者超过浮点数尾数的长度。

三、总结

在设计到精度计算时,我们尽量使用 String 类型来进行转换,而且涉及到 BigDecimal 的计算,要使用其对应方法进行计算。

四、工具类

这里封装一个 BigDecimal 工具类

public class BigDecimalUtils {
    
    public static BigDecimal doubleAdd(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.add(b2);
    }
    
    public static BigDecimal floatAdd(float v1, float v2) {
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        return b1.add(b2);
    }
    
    public static BigDecimal doubleSub(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.subtract(b2);
    }
    
    public static BigDecimal floatSub(float v1, float v2) {
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        return b1.subtract(b2);
    }
    
    public static BigDecimal doubleMul(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.multiply(b2);
    }
    
    public static BigDecimal floatMul(float v1, float v2) {
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        return b1.multiply(b2);
    }
    
    public static BigDecimal doubleDiv(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        // 保留小数点后两位 ROUND_HALF_UP = 四舍五入
        return b1.divide(b2, 2, RoundingMode.HALF_UP);
    }
    
    public static BigDecimal floatDiv(float v1, float v2) {
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        // 保留小数点后两位 ROUND_HALF_UP = 四舍五入
        return b1.divide(b2, 2, RoundingMode.HALF_UP);
    }
    
    public static int doubleCompareTo(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.compareTo(b2);
    }
    
    public static int floatCompareTo(float v1, float v2) {
        BigDecimal b1 = new BigDecimal(Float.toString(v1));
        BigDecimal b2 = new BigDecimal(Float.toString(v2));
        return b1.compareTo(b2);
    }
}

到此这篇关于Java BigDecimal正确用法详解的文章就介绍到这了,更多相关Java BigDecimal内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java BigDecimal正确用法详解

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

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

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

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

下载Word文档
猜你喜欢
  • Java BigDecimal正确用法详解
    目录一、背景二、事故案例1、问题2、问题复现3、源码分析4、原因分析三、总结四、工具类一、背景 BigDecimal 平时主要用于计算金钱时,其自身提供了很多的构造方法,但是这些构造...
    99+
    2022-11-13
  • Java精确计算BigDecimal类详解
    引言 float和double类型的主要设计目标是为了科学计算和工程计算。他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提供完...
    99+
    2022-11-12
  • Java BigDecimal类用法详解
    一、简介 Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算。双精度浮点型变量double可以处理16位有效数。在实际应用中...
    99+
    2022-11-12
  • java中BigDecimal用法详解
    首先,学习一个东西,我们都必须要带着问题去学,这边我分为 【为什么?】【是什么?】【怎么用?】 【为什么要用BigDecimal?】 首先,我们先看一下,下面这个现象 那为什么会出...
    99+
    2022-11-12
  • gitstash的正确用法详解
    目录概述git stash 具体用法概述 前段时间突然发现,我之前对git stash的使用都是错误的。 具体说来,我是这么使用的:在远端有新的提交,需要git pull来拉取合并...
    99+
    2023-02-06
    git stash用法 git stash
  • Java BigDecimal案例详解
    引言   float和double类型的主要设计目标是为了科学计算和工程计算。他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提...
    99+
    2022-11-12
  • Java BigDecimal中divide方法案例详解
    1、首先说一下用法,BigDecimal中的divide主要就是用来做除法的运算。其中有这么一个方法. public BigDecimal divide(BigDecimal d...
    99+
    2022-11-12
  • Java中Optional的正确用法与争议点详解
    目录简介#创建Optional判空取值函数式处理Optional争议点总结简介# 要说Java中什么异常最容易出现,我想NullPointerException一定当仁不让,为了解决...
    99+
    2022-11-13
    Java中Optional用法 Optional java Java中Optional类的使用方法
  • useCallback和useMemo的正确用法详解
    目录正文缓存值缓存函数组件为什么会重新渲染缓存复杂的计算什么是复杂的计算何时进行优化呢总结正文 现实很多项目存在大量的useMemo和useCallback,大多数的使用并没有起到实...
    99+
    2023-01-15
    useCallback和useMemo用法 useCallback useMemo
  • java BigDecimal类案例详解
    目录前言一.介绍二.知识点介绍三.知识点详解1、概述2、BigDecimal构造方法3、源码的描述4、BigDecimal加减乘除运算5、总结6、精炼练习6.1  题目6....
    99+
    2022-11-13
  • 详解Java中List的正确的删除方法
    目录简介实例正确方法法1:for的下标倒序遍历法2: list.stream().filter().collect()法3: iterator迭代器错误方法法1:for(xxx : ...
    99+
    2022-11-13
  • BigDecimal中divide方法详解
    BigDecimal类中的divide方法用于执行两个BigDecimal对象的除法运算。该方法有多种重载形式,下面列举其中一种常用...
    99+
    2023-09-13
    divide
  • js indexOf 的正确用法示例详解
    目录js indexOf 的正确用法补充:js中indexOf()的使用示例1:查找字符串中某一字符从头开始第一次出现的索引示例2:查找字符串中某一字符从指定位置开始第一次出现的索引...
    99+
    2023-02-05
    js indexOf 用法 js indexOf 使用
  • 正则表达式中问号(?)的正确用法详解
    目录1、直接跟随在子表达式后面2、非贪婪匹配3、非获取匹配4、断言参考资料:正则表达式中“”的用法大概有以下几种 1、直接跟随在子表达式后面 这种方式是最常用...
    99+
    2022-11-13
  • 详解java中BigDecimal精度问题
    目录一、背景二、BigDecimal构造函数1、四种构造函数2、为什么会出现这种情况3、如何解决三、常用方法1、常用方法2、取舍规则四、格式化一、背景 在实际开发中,对于 不需要任何...
    99+
    2022-11-12
  • java 中ThreadLocal 的正确用法
    java 中ThreadLocal 的正确用法用法一:在关联数据类中创建private static ThreadLocalThreaLocal的JDK文档中说明:ThreadLocal instances are typically pr...
    99+
    2023-05-31
    java threadlocal ava
  • Java精确计算BigDecimal类怎么使用
    本篇内容主要讲解“Java精确计算BigDecimal类怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java精确计算BigDecimal类怎么使用”吧!引言float和double类型...
    99+
    2023-06-25
  • 详解ubuntu安装opencv的正确方法
    本文介绍的是如何安装ubuntu下C++接口的opencv 1.安装准备: 1.1安装cmake sudo apt-get install cmake 1.2依赖环境 sudo apt-get instal...
    99+
    2022-06-04
    ubuntu安装opencv ubuntu opencv安装
  • 关于Java如何正确地实现方法重载详解
    目录一. 方法重载1. 概念2. 要求二. 代码案例1. 普通方法重载2. 构造方法重载三. 结语一. 方法重载 1. 概念 所谓方法的重(chong)载(重复出现),就是指在一个类...
    99+
    2023-05-19
    Java实现方法重载 Java方法重载
  • Protobuf在Cmake中的正确使用方法详解
    Protobuf是google开发的一个序列化和反序列化的协议库,我们可以自己设计传递数据的格式,通过.proto文件定义我们的要传递的数据格式。例如,在深度学习中常用的ONNX交换...
    99+
    2022-11-11
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作