在看《阿里巴巴开发手册》里面提到浮点数之间的等值判断不要用 ==,而是指定误差范围或用BigDecimal,然后才记忆起备忘录里BigDecimal还没写呢,就这篇幅写了一星期,因为实习完全没有时间啊啊啊啊啊啊啊啊
1. BigDecimal介绍
背景
我们知道计算机都是以二进制的形式存储数据的,而我们日常则是使用十进制,那么我们的 数字
存进计算机则需一个进制转换的过程,这过程就会损失精度的,就导致浮点数不能用等值判断
原因
十进制的0.1
转换 成二进制为 0.00011001...
,是无限小数,计算机存不下只能截取。后面这截取的无限小数还原成十进制就会损失精度不准确,不能用等值判断了
救星
BigDecimal的横空出世是为了解决浮点数的精度问题,其全限定类名为 java.math.BigDecimal,BigDecimal是一个对象,代表着不变的,任意精度的带符号的十进制数字,我们要使用该对象的方法来进行加减乘除的操作
原理
既然十进制小数转成二进制会损失精度,那么把十进制小数扩大成整数再转成二进制则会保持精度了
2. BigDecimal基本使用
2.1 常见构造方法
方法 | 描述 |
BigDecimal(int val) | 将int转换成BigDecimal |
BigDecimal(int val, MathContext mc) | 根据上下设置进行舍入 |
BigDecimal(long val) | 将long转换成BigDecimal |
BigDecimal(double) | 将double转换成BigDecimal |
BigDecimal(String) | 将String转换成BigDecimal |
MathContext(int setPrecision, RoundingMode setRoundingMode) | 上下文取舍(精度,舍入模式) |
阿里手册规约:禁止使用构造方法 BigDecimal(double)的方式把 double 值转化为 BigDecimal 对象
BigDecimal num1 = new BigDecimal(0.1);
BigDecimal num2 = new BigDecimal("0.1");
System.out.println(num1); // 0.1000000000000000055511151231257827021181583404541015625
System.out.println(num2); // 0.1
因为浮点数并不是一个准确的值,而String类型的就确定
2.2 常用方法
方法 | 描述 |
abs() | 返回一个绝对值BigDecimal对象 |
scale() | 小数位数,包含末尾零。返回负数表示是一个正数,且有负数位0 |
stripTrailingZeros() | 除去末尾的零,包含整数 |
add(BigDecimal augend) | 被加数 |
add(BigDecimal augend, MathContext mc) | 根据上下文取舍 |
subtract(BigDecimal subtrahend) | 被减数 |
multiply(BigDecimal multiplicand) | 被乘数 |
divide(BigDecima divisor, int roundingMode) | 被除数,要指定上下文取舍,否则报错 |
divideAndRemainder(BigDecimal divisor) | 求余 |
compareTo(BigDecimal val) | 比较数值大小,equals要求scale()相同,且值相同 |
toString() | 转成字符串 |
intValue() | 转成整型 |
longValue() | 转成长整型 |
BigDecimal num1 = new BigDecimal("0.01234");
BigDecimal num2 = new BigDecimal("0.56789");
-------------------------------------------------------------------
System.out.println(num1.add(num2));
System.out.println(num1.subtract(num2));
// 小数保留6位,四舍五入
System.out.println(num1.multiply(num2,new MathContext(6, RoundingMode.HALF_UP)));
// 小数保留4位,直接截取
System.out.println(num1.divide(num2,new MathContext(4,RoundingMode.DOWN)));
// 0.58023
// -0.55555
// 0.00700776
// 0.02172
-------------------------------------------------------------------
BigDecimal num1 = new BigDecimal("0.123");
BigDecimal num2 = new BigDecimal("0.123000");
System.out.println(num1.equals(num2));
System.out.println(num1.compareTo(num2));
// false
// 0
除法存在除不尽的情况,所以一定要使用上下文取舍器
BigDecimal比较用compareTo,而equals要求scale()即小数位数相同
ArithmeticException常见的算数异常
3. BigInteger
Java原生提供的最大整型是长整型,占8字节64位,范围是-9223372036854775808 ~ 9223372036854775807,如果超过了这个范围,那么可以用不可变的BigInteger对象,其原理是内部使用 int[] 数组来模拟大数
3.1 常见构造函数
函数 | 描述 |
BigInteger(byte[] val) | |
BigInteger(String) val) |
3.2 常见方法
方法 | 描述 |
add(BigInteger) val) | 加法 |
subtract(BigInteger val) |
BigInteger num1 = new BigInteger("1234567890");
System.out.println(num1.pow(5)); // 2867971860299718107233761438093672048294900000
BigInteger num2 = new BigInteger("123456");
long num3 = num2.longValue();
System.out.println(num3); // 123456