【Java SE】| BigDecimal妙用

这篇具有很好参考价值的文章主要介绍了【Java SE】| BigDecimal妙用。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一. 🦁 前言

在使用浮点数(float / double)进行运算时,会发生精度丢失的风险(因为无限循环小数无法使用二进制来精确表示,会被截断,精度丢失)。所以我们要用到BigDecimal,其可以实现对浮点数的运算,不会造成精度丢失

二. 🦁 具体使用

1. 创建

避免使用float或double类型作为BigDecimal的构造参数,因为这样会导致精度丢失。应该使用字符串类型或使用BigDecimal类的静态方法valueOf来创建BigDecimal对象

《阿里巴巴Java开发手册》里面强调:禁止使用构造方法BigDecimal(double)来构建对象,优先推荐入参方法为String的构造方法创建,或者使用BigDecimal.valueOf()静态方法来转化。所以正确的创建方法为:

new BigDecimal("0.10")
or
BigDecimal.valueOf(0.10);

2. 加减乘除

  • add 方法用于将两个 BigDecimal 对象相加;
  • subtract 方法用于将两个 BigDecimal 对象相减;
  • multiply 方法用于将两个 BigDecimal 对象相乘;
  • divide 方法用于将两个 BigDecimal 对象相除。
BigDecimal a = new BigDecimal("1.0");
BigDecimal b = new BigDecimal("0.9");
System.out.println(a.add(b));// 1.9
System.out.println(a.subtract(b));// 0.1
System.out.println(a.multiply(b));// 0.90
System.out.println(a.divide(b));// 无法除尽,抛出 ArithmeticException 异常
System.out.println(a.divide(b, 2, RoundingMode.HALF_UP));// 1.11

这里的除法是以三个变量为参数的,第二个参数表示保留多少位小数,第三个参数表示保留规则。

3. 大小比较

a.compareTo(b) : 返回 -1 表示 a 小于 b,0 表示 a 等于 b , 1 表示 a 大于 b。

BigDecimal a = new BigDecimal("1.0");
BigDecimal b = new BigDecimal("0.9");
System.out.println(a.compareTo(b));// 1

4. 保留几位小数

通过 setScale方法设置保留几位小数以及保留规则。

BigDecimal m = new BigDecimal("1.255433");
BigDecimal n = m.setScale(3,RoundingMode.HALF_DOWN);
System.out.println(n);// 1.255

5. BigDecimal 等值比较问题

tips:
在使用BigDecimal进行等值比较时,需要注意以下几点:

  • 使用equals方法进行比较,不能使用"==“或”!=",因为BigDecimal是对象,而不是基本数据类型。
  • 确保比较的精度一致,可以使用compareTo方法进行比较

【Java SE】| BigDecimal妙用

BigDecimal 使用 equals() 方法进行等值比较出现问题的代码示例:

BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("1.0");
System.out.println(a.equals(b));//false

这是因为 equals() 方法不仅仅会比较值的大小(value)还会比较精度(scale),而 compareTo() 方法比较的时候会忽略精度

1.0 的 scale 是 1,1 的 scale 是 0,因此 a.equals(b) 的结果是 false。如下:
【Java SE】| BigDecimal妙用

compareTo() 方法可以比较两个 BigDecimal 的值,如果相等就返回 0,如果第 1 个数比第 2 个数大则返回 1,反之返回-1

BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("1.0");
System.out.println(a.compareTo(b));//0

三. 🦁 BigDecimal工具类

开发中,为了更方便使用BigDecimal,将其封装成一个个工具类最好。

import java.math.BigDecimal;
import java.math.RoundingMode;

/**
 * 简化BigDecimal计算的小工具类
 */
public class BigDecimalUtil {

    /**
     * 默认除法运算精度
     */
    private static final int DEF_DIV_SCALE = 10;

    private BigDecimalUtil() {
    }

    /**
     * 提供精确的加法运算。
     *
     * @param v1 被加数
     * @param v2 加数
     * @return 两个参数的和
     */
    public static double add(double v1, double v2) {
        BigDecimal b1 = BigDecimal.valueOf(v1);
        BigDecimal b2 = BigDecimal.valueOf(v2);
        return b1.add(b2).doubleValue();
    }

    /**
     * 提供精确的减法运算。
     *
     * @param v1 被减数
     * @param v2 减数
     * @return 两个参数的差
     */
    public static double subtract(double v1, double v2) {
        BigDecimal b1 = BigDecimal.valueOf(v1);
        BigDecimal b2 = BigDecimal.valueOf(v2);
        return b1.subtract(b2).doubleValue();
    }

    /**
     * 提供精确的乘法运算。
     *
     * @param v1 被乘数
     * @param v2 乘数
     * @return 两个参数的积
     */
    public static double multiply(double v1, double v2) {
        BigDecimal b1 = BigDecimal.valueOf(v1);
        BigDecimal b2 = BigDecimal.valueOf(v2);
        return b1.multiply(b2).doubleValue();
    }

    /**
     * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到
     * 小数点以后10位,以后的数字四舍五入。
     *
     * @param v1 被除数
     * @param v2 除数
     * @return 两个参数的商
     */
    public static double divide(double v1, double v2) {
        return divide(v1, v2, DEF_DIV_SCALE);
    }

    /**
     * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
     * 定精度,以后的数字四舍五入。
     *
     * @param v1    被除数
     * @param v2    除数
     * @param scale 表示表示需要精确到小数点以后几位。
     * @return 两个参数的商
     */
    public static double divide(double v1, double v2, int scale) {
        if (scale < 0) {
            throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
        }
        BigDecimal b1 = BigDecimal.valueOf(v1);
        BigDecimal b2 = BigDecimal.valueOf(v2);
        return b1.divide(b2, scale, RoundingMode.HALF_UP).doubleValue();
    }

    /**
     * 提供精确的小数位四舍五入处理。
     *
     * @param v     需要四舍五入的数字
     * @param scale 小数点后保留几位
     * @return 四舍五入后的结果
     */
    public static double round(double v, int scale) {
        if (scale < 0) {
            throw new IllegalArgumentException(
                    "The scale must be a positive integer or zero");
        }
        BigDecimal b = BigDecimal.valueOf(v);
        BigDecimal one = new BigDecimal("1");
        return b.divide(one, scale, RoundingMode.HALF_UP).doubleValue();
    }

    /**
     * 提供精确的类型转换(Float)
     *
     * @param v 需要被转换的数字
     * @return 返回转换结果
     */
    public static float convertToFloat(double v) {
        BigDecimal b = new BigDecimal(v);
        return b.floatValue();
    }

    /**
     * 提供精确的类型转换(Int)不进行四舍五入
     *
     * @param v 需要被转换的数字
     * @return 返回转换结果
     */
    public static int convertsToInt(double v) {
        BigDecimal b = new BigDecimal(v);
        return b.intValue();
    }

    /**
     * 提供精确的类型转换(Long)
     *
     * @param v 需要被转换的数字
     * @return 返回转换结果
     */
    public static long convertsToLong(double v) {
        BigDecimal b = new BigDecimal(v);
        return b.longValue();
    }

    /**
     * 返回两个数中大的一个值
     *
     * @param v1 需要被对比的第一个数
     * @param v2 需要被对比的第二个数
     * @return 返回两个数中大的一个值
     */
    public static double returnMax(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(v1);
        BigDecimal b2 = new BigDecimal(v2);
        return b1.max(b2).doubleValue();
    }

    /**
     * 返回两个数中小的一个值
     *
     * @param v1 需要被对比的第一个数
     * @param v2 需要被对比的第二个数
     * @return 返回两个数中小的一个值
     */
    public static double returnMin(double v1, double v2) {
        BigDecimal b1 = new BigDecimal(v1);
        BigDecimal b2 = new BigDecimal(v2);
        return b1.min(b2).doubleValue();
    }

    /**
     * 精确对比两个数字
     *
     * @param v1 需要被对比的第一个数
     * @param v2 需要被对比的第二个数
     * @return 如果两个数一样则返回0,如果第一个数比第二个数大则返回1,反之返回-1
     */
    public static int compareTo(double v1, double v2) {
        BigDecimal b1 = BigDecimal.valueOf(v1);
        BigDecimal b2 = BigDecimal.valueOf(v2);
        return b1.compareTo(b2);
    }

}

四. 🦁 项目实战应用场景

BigDecimal 是 Java 中一个用于高精度计算的类,它能够处理浮点数的精度问题,避免了由于计算机精度问题带来的计算错误。在实际的开发中,BigDecimal 通常用于以下情况:

1.金融计算
在金融领域,精度要求非常高,一丝一毫的误差都可能导致巨大的损失。BigDecimal 在处理货币计算时,能够确保计算结果精确。例如,计算两个金额的差异,或者计算税收,利息,折扣等等。

2.科学计算
在科学计算领域,对数值计算精度的要求也非常高。因为在科学计算中,数据量通常都很大,浮点数精度问题会导致计算结果的偏差。BigDecimal 不仅可以处理大量数据,而且能够确保计算结果的精度。

3.商业应用
在商业应用中,经常需要处理大量的数据,如订单金额,销售统计等等,因此,使用 BigDecimal 可以确保计算结果的精度不会受到机器精度的限制,从而提高数据处理的准确性。

4.工程计算
在工程计算领域,有时需要计算非常准确的计算结果,例如,计算房屋建筑面积、体积等。由于 BigDecimal 能够处理小数点以后的位数,因此,能够确保计算结果的精度。

综上所述,BigDecimal 在处理需要高精度计算的场景中,非常适用。在实际开发中,开发者应该根据实际需求,灵活运用 BigDecimal,确保计算结果的准确性。

【Java SE】| BigDecimal妙用文章来源地址https://www.toymoban.com/news/detail-470554.html

到了这里,关于【Java SE】| BigDecimal妙用的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java中BigDecimal保留两位小数

    方案一: 方案二:#0.00% 表示保留两位小数,.后几个零就是保留几位小数 %号表示会转成百分率。 方案三: 方案四:保留两位小数,不四舍五入 金额以万元展示,保留两位小数 Bigdecimal 通常用来表示和金额相关的数字。 常用的方法有: BigDecimal a. add (BigDecimal b) BigDecimal对象

    2024年02月09日
    浏览(33)
  • java实现BigDecimal/int四舍五入

    输出结果: 四舍五入(保留位后一位是5进位) = 2.4 四舍五入(保留位后一位是5舍弃) = 2.3 四舍五入(保留位是奇数) = 4.6 四舍五入(保留位是偶数) = 4.4 进位 = 2.4 说明: ROUND_CEILING Rounding mode to round towards positive infinity. 向正无穷方向舍入 ROUND_DOWN Rounding mode to round towards zero. 向零方向

    2024年02月09日
    浏览(32)
  • Java 数学计算(一):BigDecimal 详解 + 代码示例

    ❤️ 个人主页:水滴技术 🌸 订阅专栏:Java 教程:从入门到精通 🚀 支持水滴: 点赞 👍 + 收藏 ⭐ + 留言 💬 大家好,我是水滴~~ 本文主要对 BigDecimal 类进行详细介绍,主要内容有:构造器、静态方法、静态常量、常用方法、舍入模式,并提供了大量的代码示例,可以将该

    2023年04月08日
    浏览(28)
  • 深入了解:Java中BigDecimal比较大小的方法

    目录 Java中BigDecimal比较大小的方法 1. ​​compareTo()​​ 2. ​​equals()​​ 3. ​​compareTo()​​ 大家好,今天我想和大家讨论一下Java中BigDecimal类的比较大小的方法。在实际开发中,我们经常会遇到需要比较两个BigDecimal对象的大小的情况,比如排序、判断大小等。但是由于Bi

    2024年02月04日
    浏览(26)
  • 日常BUG——Java使用Bigdecimal类型报错

    😜 作           者 :是江迪呀 ✒️ 本文 : 日常BUG 、 BUG 、 问题分析 ☀️ 每日   一言 : 存在错误说明你在进步! 直接上代码: 报错内容: 原来JAVA中如果用 BigDecimal 做除法的时候一定要在divide方法中传递第二个参数,定义精确到小数点后几位,否则在不整除

    2024年02月13日
    浏览(34)
  • Java之BigDecimal系列--去掉小数末尾多余的0

    原文网址:Java之BigDecimal系列--去掉小数末尾多余的0_IT利刃出鞘的博客-CSDN博客 说明 本文介绍Java去掉BigDecimal小数末尾多余的0的方法。 概述 BigDecimal提供了stripTrailingZeros()方法可以实现去掉小数末尾的 0。 调用了stripTrailingZeros()再调用 toString() 将会转换为科学计数法输出。如果

    2024年02月11日
    浏览(28)
  • Java进行数字计算 BigDecimal计算(加减乘除)

    Double只能处理16位有效数精度,在某些情况下精度对其需求是不够的,所以就有了BigDecimal。因为BigDecimal的精度范围的范围大,所以在问我们的开发业务中对精度要求高的属性,就需要BigDecimal来进行存储计算,防止精度丢失。这里我主要介绍一下BigDecimal的加,减,乘,除。四

    2023年04月08日
    浏览(29)
  • java BigDecimal使用(小数点,字符串等转换)

    Java中提供了操作大数字(超过 16位 有效位)的类, 即 java.math.BigInteger 类和 java.math.BigDecimal 类,用于高精度计算。 float和Double只能用来做科学计算、工程计算等;在商业计算中,对数字精度要求较高(例如货币值),必须使用 BigInteger 类和 BigDecimal 类,它支持任何精度的定点数,

    2024年02月14日
    浏览(30)
  • java stream实现分组BigDecimal求和,自定义分组求和

    随着微服务的发展,越来越多的sql处理被放到java来处理,数据库经常会使用到对集合中的数据进行分组求和,分组运算等等。 那怎么样使用java的stream优雅的进行分组求和或运算呢? 这里测试数据学生,年龄类型是Integer,身高类型是BigDecimal,我们分别对身高个年龄进行求和

    2024年02月01日
    浏览(37)
  • GMP库使用以及java中的BigInteger和BigDecimal

    一、GMP介绍和安装 GMP library全称是GNU Multiple Precision Arithmetic Library,即GNU高精度算术运算库。在网络安全技术领域中各种加密算法的软件实现始终有一个共同话题是如何在普通的PC机上实现大数运算。普通的PC机内部字长最多时32位或64位,但各种加密算法中为了达到一定安全强

    2024年02月08日
    浏览(34)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包