原文地址:https://blog.csdn.net/cen_s/article/details/76472834

在日常开发中我们经常会碰到小数计算,而小数直接计算的话会出现一些小小的错误,如下

  1. System.out.println(1.01 + 2.02);

你说能输出什么?3.03?实际上输出的是3.0300000000000002。这是因为不论是float 还是double都是浮点数,而计算机是二进制的,浮点数会失去一定的精确度。有没有不失精度的办法呢?这里就要用到BigDecimal了。

java.math.BigDecimal。Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算。双精度浮点型变量double可以处理16位有效数。在实际应用中,需要对更大或者更小的数进行运算和处理。float和double只能用来做科学计算或者是工程计算,在商业计算中要用java.math.BigDecimal。BigDecimal所创建的是对象,我们不能使用传统的+、-、*、/等算术运算符直接对其对象进行数学运算,而必须调用其相对应的方法。方法中的参数也必须是BigDecimal的对象。构造器是类的特殊方法,专门用来创建对象,特别是带有参数的对象。

创建BigDecimal对象主要有两种。

  1. BigDecimal b1 = new BigDecimal("1.34");//1.34
  2. BigDecimal b2 = BigDecimal.valueOf(1.34);//1.34

其中b1也可以写成new BigDecimal(Double.toString(1.34)),可以直接new BigDecimal(1.34)吗,也是可以的,只是会出现上述的精度问题。

  1. BigDecimal one1 = new BigDecimal(1.34);//1.3400000000000000799360577730112709105014801025390625
  2. BigDecimal two1 = new BigDecimal("1.34");//1.34

除了这两种外,特殊的像0、1、10可以这样写。

  1. BigDecimal zero = BigDecimal.ZERO;
  2. BigDecimal one = BigDecimal.ONE;
  3. BigDecimal ten = BigDecimal.TEN;

比较一下BigDecimal.ZERO、new BigDecimal("0")、BigDecimal.valueOf(0)这三者,equals都是true,==的话new BigDecimal("0")就不用看了都new了,而BigDecimal.ZERO
== BigDecimal.ZERO为true。查看一下源代码可得。

  1. public static BigDecimal valueOf(long val) {
  2. if (val >= 0 && val < zeroThroughTen.length)
  3. return zeroThroughTen[(int)val];
  4. else if (val != INFLATED)
  5. return new BigDecimal(null, val, 0, 0);
  6. return new BigDecimal(INFLATED_BIGINT, val, 0, 0);
  7. }
  1. // Cache of common small BigDecimal values.
  2. private static final BigDecimal zeroThroughTen[] = {
  3. new BigDecimal(BigInteger.ZERO, 0, 0, 1),
  4. new BigDecimal(BigInteger.ONE, 1, 0, 1),
  5. new BigDecimal(BigInteger.valueOf(2), 2, 0, 1),
  6. new BigDecimal(BigInteger.valueOf(3), 3, 0, 1),
  7. new BigDecimal(BigInteger.valueOf(4), 4, 0, 1),
  8. new BigDecimal(BigInteger.valueOf(5), 5, 0, 1),
  9. new BigDecimal(BigInteger.valueOf(6), 6, 0, 1),
  10. new BigDecimal(BigInteger.valueOf(7), 7, 0, 1),
  11. new BigDecimal(BigInteger.valueOf(8), 8, 0, 1),
  12. new BigDecimal(BigInteger.valueOf(9), 9, 0, 1),
  13. new BigDecimal(BigInteger.TEN, 10, 0, 2),
  14. };

发现10以内的对象都是同一个,所以为true。

BigDecimal的加减乘除运算。

  1. public BigDecimal add(BigDecimal value);//加法
  2. public BigDecimal subtract(BigDecimal value);//减法
  3. public BigDecimal multiply(BigDecimal value);//乘法
  4. public BigDecimal divide(BigDecimal value);//除法

也可以照下面加法例子写成一个util,另外三个都差不多就不展开了。

  1. public static double add(double value1,double value2){
  2. BigDecimal b1 = new BigDecimal(Double.toString(value1));
  3. BigDecimal b2 = new BigDecimal(Double.toString(value2));
  4. return b1.add(b2).doubleValue();
  5. }

BigDecimal的运算都没有对原值进行操作,而是返回一个新的BigDecimal对象,这点可能有些小伙伴会搞错要注意一下。

  1. BigDecimal b1 =new BigDecimal("1.34");
  2. System.out.println("b1: " + b1);
  3. BigDecimal b2 =new BigDecimal("2.34");
  4. b1.add(b2);
  5. System.out.println("b1: " + b1);//b1并没有变

BigDecimal的比较用的是BigDecimal的compareTo方法,将此 BigDecimal 与指定的 BigDecimal 比较。

根据此方法,值相等但具有不同标度的两个BigDecimal对象(如,2.0 和 2.00)被认为是相等的。

当此 BigDecimal 在数字上小于、等于或大于被比较对象时,返回 -1、0 或 1。

    1. BigDecimal one = BigDecimal.valueOf(1);
    2. BigDecimal two = BigDecimal.valueOf(2);
    3. BigDecimal three = one.add(two);
    4. int i1 = one.compareTo(two);//-1
    5. int i2 = two.compareTo(two);//0
    6. int i3 = three.compareTo(two);//1

[转]Java中BigDecimal的使用的更多相关文章

  1. Java中BigDecimal的8种舍入模式是怎样的

    Java中BigDecimal的8种舍入模式是怎样的?下面长沙欧柏泰克软件学院和大家一起来学习下吧:  java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecima ...

  2. 关于java中BigDecimal的简介

    关于java中BigDecimal的简介 1.BigDecimal属于大数据,精度极高,不属于基本数据类型,属于java对象(引用数据类型), 这是sun提供的一个类,专门用在财务软件中. 2.注意: ...

  3. java中BigDecimal加减乘除基本用法

    Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算.双精度浮点型变量double可以处理16位有效数. 在实际应用中,需要对更大或者更小的数进 ...

  4. Java中BigDecimal的8种舍入模式

    java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecimal 由任意精度的整数非标度值和32位的整数标度(scale)组成. 如果为零或正数,则标度是小数点后的位 ...

  5. Java中BigDecimal的舍入模式

    java.math.BigDecimal 不可变的.任意精度的有符号十进制数.BigDecimal 由任意精度的整数非标度值和32位的整数标度(scale)组成. 如果为零或正数,则标度是小数点后的位 ...

  6. Java中BigDecimal类介绍及用法

    Java中提供了大数字(超过16位有效位)的操作类,即 java.math.BinInteger 类和 java.math.BigDecimal 类,用于高精度计算. 其中 BigInteger 类是 ...

  7. java中BigDecimal在金融行业中的使用

    1.引言 在java语言中,double和float用于二进制浮点型计算,无法得到精确的结果.而BigDecimal则用于精确的计算.不超过16位有效数字(最好是不超过13位)的科学和工程计算,可以使 ...

  8. java中BigDecimal的学习

    干着java的活,但是看的都是一些偏底层的东西(或者我根本就没有看),有点荒废了java的学习. 最近一直在用到一个类是BigDecimal,但都是模棱两可地在那儿用,并没有深入研究这个类的细节,感觉 ...

  9. Java基础扫盲系列(二)—— Java中BigDecimal和浮点类型

    一直以来我几乎未使用过BigDecimal类型,只有在DB中涉及到金额字段时听说要用Decimal类型,但是今天再项目代码中看到使用BigDecimal表示贷款金额. 本篇文章不是介绍BigDecim ...

随机推荐

  1. C# 进制转换(二进制、十六进制、十进制互转)

    原文地址:https://www.cnblogs.com/icebutterfly/p/8884023.html C# 进制转换(二进制.十六进制.十进制互转)由于二进制数在C#中无法直接表示,所以所 ...

  2. C# 禁止datagridview 自动产生列

    dataGridView1.AutoGenerateColumns = false;

  3. Oracle 12C -- 设置CDB启动后,PDBs自动启动

    CDB重启后,PDBs默认是处于mounted状态 SQL> select name,open_mode from v$pdbs; NAME OPEN_MODE ---------------- ...

  4. MFC动态按钮的创建及其消息响应 和 自定义消息

    原文链接: http://www.cnblogs.com/gaohongchen01/p/4046525.html 动态按钮(多个)的创建: 1.在类中声明并定义按钮控件的ID #define IDC ...

  5. Python小游戏、小程序

    python 小游戏之摇骰子猜大小 python 实现一个双色球生成程序 python-循环与判断练习题

  6. 解决iPad/iPhone下手机号码会自动被加上a标签的问题

    解决办法: 在页面顶部<head></head>中增加: <meta name="format-detection" content="te ...

  7. 持续集成(1)gitlab的安装

    操作系统:centos 6.5 关闭selinux # 修改/etc/selinux/config 文件 将SELINUX=enforcing改为SELINUX=disabled ,然后重启电脑 # ...

  8. css animation和keyframes

    keyframes应用在animation上,animation应用在元素上. <html> <style type="text/css"> .div1 { ...

  9. 使用 Apache Commons CLI 开发命令行工具示例

    Apache Commons CLI 简介 Apache Commons CLI 是 Apache 下面的一个解析命令行输入的工具包,该工具包还提供了自动生成输出帮助文档的功能. Apache Com ...

  10. javascript读取xml文件读取节点数据的例子

    分享下用javascript读取xml文件读取节点数据方法. 读取的节点数据,还有一种情况是读取节点属性数据. <head> <title></title> < ...