java中针对BigDecimal 类型字段处理

本文深入讲解了Java中BigDecimal类的初始化方法、四则运算操作、以及如何处理除法中的无限循环小数问题。特别强调了使用String而非double初始化BigDecimal的重要性,避免精度损失。同时,文章还介绍了BigDecimal的setScale方法用于保留指定位数的小数,divide方法的正确使用,以及compareTo方法用于比较大小。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

初始化,加减乘除

初始化

初始化0

//建议使用
BigDecimal bignum1 = BigDecimal.ZERO;

//或者
BigDecimal bignum1 = new BigDecimal(0);

//或者
BigDecimal bignum1 = new BigDecimal("0");

初始化int

//建议使用
BigDecimal bignum1 = new BigDecimal("10000");

//或者
BigDecimal bignum1 = new BigDecimal(10000);

初始化double

不能使用double类型来初始化BigDecimal类型,如:BigDecimal bignum1 = new BigDecimal(0.06);

要使用String类型的数值字符串来初始化,如:BigDecimal bignum1=new BigDecimal("0.06");

因为:BigDecimal(double val)这个构造方法有时是无法精确预料的

传入0.1,有可能变成0.1000000000000000055511151231257827021181583404541015625。

因为double类型无法精确地存储0.1

实例:

public class BigDecimalInitTest {

public static void main(String[] args) {

BigDecimal bignum1=new BigDecimal("0.06");

BigDecimal bignum2=new BigDecimal(0.06);

System.out.println(bignum1);

System.out.println(bignum2);

}

}

运行之后,结果为:

0.06

0.059999999999999997779553950749686919152736663818359375

加 add

BigDecimal bignum1 = new BigDecimal("10");  
BigDecimal bignum2 = new BigDecimal("5");  
System.out.println("和 是:" + bignum1.add(bignum2)); 

减 subtract

BigDecimal bignum1 = new BigDecimal("10");  
BigDecimal bignum2 = new BigDecimal("5");  
System.out.println("差  是:" + bignum1.subtract(bignum2)); 

乘 multiply

BigDecimal bignum1 = new BigDecimal("10");  
BigDecimal bignum2 = new BigDecimal("5");  
System.out.println("积  是:" + bignum1.multiply(bignum2)); 

除 divide

BigDecimal bignum1 = new BigDecimal("10");  
BigDecimal bignum2 = new BigDecimal("5");  
System.out.println("商  是:" + bignum1.divide(bignum2)); 

四舍五入后保留两位小数 setScale

BigDecimal bignum1 = new BigDecimal("10.2536");  
System.out.println("四舍五入后保留两位小数 是:" + bignum1.setScale(2, BigDecimal.ROUND_HALF_UP)); 

除 divide,四舍五入后保留两位小数

BigDecimal bignum1 = new BigDecimal("1450.2536");
BigDecimal bignum2 = new BigDecimal(100);
System.out.println("四舍五入后保留两位小数 是:" + bignum1.divide(bignum2, 2, BigDecimal.ROUND_HALF_UP));

BigDecimal的divide方法报错

java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.

报错语句:

a.divide(b);

原因:

JAVA中如果用BigDecimal做除法的时候一定要在divide方法中传递第二个参数,定义精确到小数点后几位,否则在不整除的情况下,结果是无限循环小数时,就会抛出以上异常。

解决方法:

a.divide(b, 2, BigDecimal.ROUND_HALF_UP);
四舍五入:BigDecimal.ROUND_HALF_UP

比较大小

compareTo方法

compareTo方法会返回一个整数值,该值指示一个BigDecimal相对于另一个BigDecimal的顺序。如果这个BigDecimal等于另一个BigDecimal,则返回0;如果这个BigDecimal小于另一个BigDecimal,则返回-1;如果这个BigDecimal大于另一个BigDecimal,则返回1。

以下是使用compareTo方法比较两个BigDecimal对象大小的示例代码:

import java.math.BigDecimal;
 
public class BigDecimalComparison {
    public static void main(String[] args) {
        BigDecimal a = new BigDecimal("1.00");
        BigDecimal b = new BigDecimal("1.0");
 
        int comparisonResult = a.compareTo(b);
 
        if (comparisonResult < 0) {
            System.out.println("a is less than b");
        } else if (comparisonResult > 0) {
            System.out.println("a is greater than b");
        } else {
            System.out.println("a is equal to b");
        }
    }
}

忽略精度进行比较

如果你想忽略精度的不同,你可能需要先通过stripTrailingZeros方法移除末尾多余的零,然后再进行比较。

BigDecimal a = new BigDecimal("1.00").stripTrailingZeros();
BigDecimal b = new BigDecimal("1.0").stripTrailingZeros();
import java.math.BigDecimal;
 
public class BigDecimalComparison {
    public static void main(String[] args) {
        BigDecimal a = new BigDecimal("1.00").stripTrailingZeros();
        BigDecimal b = new BigDecimal("1.0").stripTrailingZeros();
 
        int comparisonResult = a.compareTo(b);
 
        if (comparisonResult < 0) {
            System.out.println("a is less than b");
        } else if (comparisonResult > 0) {
            System.out.println("a is greater than b");
        } else {
            System.out.println("a is equal to b");
        }
    }
}

### MongoDB 中 BigDecimal 类型的存储与操作 #### 存储方式 在 MongoDB 中,`BigDecimal` 并不是一个原生支持的数据类型。通常情况下,Java 的 `BigDecimal` 对象会被序列化为字符串或者双精度浮点数(Double)。然而,这种转换可能会带来一些问题,比如精度丢失或无法精确表示某些数。为了确保高精度计算的需求被满足,推荐将 `BigDecimal` 转换为 BSON Decimal128 类型进行存储[^5]。 BSON Decimal128 是一种固定长度的 128 位十进制浮点数格式,能够提供更高的精度和范围,非常适合金融领域或其他需要高度准确性的地方。当使用 Java 驱动程序时,可以通过自定义编码解码逻辑来实现 `BigDecimal` 和 BSON Decimal128 之间的映射。 #### 插入数据示例 以下是插入包含 `BigDecimal` 数据的例子: ```java import org.bson.codecs.configuration.CodecRegistries; import org.bson.codecs.pojo.PojoCodecProvider; import com.mongodb.MongoClientSettings; import com.mongodb.client.MongoClients; import com.mongodb.client.MongoCollection; import com.mongodb.client.MongoDatabase; import org.bson.types.Decimal128; public class BigDecimalExample { public static void main(String[] args) { // 创建 MongoClient 连接设置 var codecRegistry = CodecRegistries.fromProviders(PojoCodecProvider.builder().automatic(true).build()); var settings = MongoClientSettings.builder() .codecRegistry(codecRegistry) .build(); try (var mongoClient = MongoClients.create(settings)) { MongoDatabase database = mongoClient.getDatabase("test"); MongoCollection<Document> collection = database.getCollection("bigDecimalCollection"); // 定义一个 BigDecimal 并将其转为 Decimal128 java.math.BigDecimal bigDecimalValue = new java.math.BigDecimal("123456789.123456789"); Document document = new Document("value", new Decimal128(bigDecimalValue)); // 插入文档 collection.insertOne(document); System.out.println("Data inserted successfully."); } } } ``` 上述代码展示了如何创建一个带有 `BigDecimal` 字段的文档,并将其作为 Decimal128 类型写入数据库中[^6]。 #### 查询数据示例 查询具有 `BigDecimal` 类型字段的内容可以按照如下方式进行: ```java // 构造查询条件 java.math.BigDecimal queryValue = new java.math.BigDecimal("123456789.123456789"); FindIterable<Document> results = collection.find(eq("value", new Decimal128(queryValue))); // 输出结果 results.forEach(doc -> System.out.println(doc.toJson())); ``` 此部分说明了基于特定查找记录的方法,其中同样涉及到了 Decimal128 的运用[^7]。 #### 性能考量 如果集合规模非常庞大,则应考虑建立适当索引来优化性能。例如,在经常按某个大数列筛选的情况下,可针对该列构建索引以减少全表扫描带来的负担[^1]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值