-
:简介
-
二:BigDecimal
-
三:工具类
-
四 :BigDecimal除法四舍五入的问题
-
五:测试
-
六 :注意的问题
一:简介
Java中的简单浮点数类型float和double不能够进行运算,或者运算会丢失精度,不光是Java,在其它很多编程语言中也有这样的问题。在大多数情况下,计算的结果是准确的,float和double只能用来做科学计算或者是工程计算,在商业计算中我们要用 java.math.BigDecimal。BigDecimal类的主要功能是进行小数的大数计算,而且最重要的是可以精确到指定的四舍五入位数。
// 浮点型计算不准确示例
@Test
public void testDouble(){
// 0.060000000000000005
System.out.println(0.05+0.01);
// 0.5800000000000001
System.out.println(1.0-0.42);
// 401.49999999999994
System.out.println(4.015*100);
// 1.2329999999999999
System.out.println(123.3/100);
// 4.01 四舍五入保留两位
System.out.println(Math.round(4.015*100)/100.0);
}
二:BigDecimal
BigDecimal有多种构造函数,常用的有2种,其中有一种不建议使用,就是double构造方式,建议使用String构造方式。
// 强制使用String的构造函数,double也有可能计算不太准确
// 原则是使用BigDecimal并且一定要用String来够造。
public BigDecimal(String val);
public BigDecimal(double val);
BigDecimal也定义了几个常用的值,0、1、10,静态的,可以通过类名直接引用BigDecimal.ZERO
public static final BigDecimal ZERO = zeroThroughTen[0];
/**
* The value 1, with a scale of 0.
*
* @since 1.5
*/
public static final BigDecimal ONE = zeroThroughTen[1];
/**
* The value 10, with a scale of 0.
*
* @since 1.5
*/
public static final BigDecimal TEN = zeroThroughTen[10];
三:工具类
由于构造方法要用String对应的构造方法,如果我们要做一个加法运算,需要先将两个浮点数转为String,然后够造成BigDecimal,在其中一个上调用add方法,传入另一个作为参数,然后把运算的结果(BigDecimal)再转换为浮点数。你能够忍受这么烦琐的过程吗?下面我们提供一个工具类来简化操作。
import java.math.BigDecimal;
/**
* 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精
* 确的浮点数运算,包括加减乘除和四舍五入。
*/
public class ArithUtil {
//默认除法运算精度
private static final int DEF_DIV_SCALE = 10;
/**
* 提供精确的加法运算。
* @param v1 被加数
* @param v2 加数
* @return 两个参数的和
*/
public static double add(double v1,double v2){
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.add(b2).doubleValue();
}
/**
* 提供精确的减法运算。
* @param v1 被减数
* @param v2 减数
* @return 两个参数的差
*/
public static double sub(double v1,double v2){
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.subtract(b2).doubleValue();
}
/**
* 提供精确的乘法运算。
* @param v1 被乘数
* @param v2 乘数
* @return 两个参数的积
*/
public static double mul(double v1,double v2){
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.multiply(b2).doubleValue();
}
/**
* 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到
* 小数点以后10位,以后的数字四舍五入。
* @param v1 被除数
* @param v2 除数
* @return 两个参数的商
*/
public static double div(double v1,double v2){
return div(v1,v2,DEF_DIV_SCALE);
}
/**
* 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
* 定精度,以后的数字四舍五入。
* @param v1 被除数
* @param v2 除数
* @param scale 表示表示需要精确到小数点以后几位。
* @return 两个参数的商
*/
public static double div(double v1,double v2,int scale){
if(scale<0){
throw new IllegalArgumentException("The scale must be a positive integer or zero");
}
BigDecimal b1 = new BigDecimal(Double.toString(v1));
BigDecimal b2 = new BigDecimal(Double.toString(v2));
return b1.divide(b2,scale,BigDecimal.ROUND_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 = new BigDecimal(Double.toString(v));
BigDecimal one = new BigDecimal("1");
return b.divide(one,scale, BigDecimal.ROUND_HALF_UP).doubleValue();
}
}
四:BigDecimal除法四舍五入
参考:https://www.cnblogs.com/shenwen/p/9481856.html
BigDecimal类的主要功能是进行小数的大数计算,而且最重要的是可以精确到指定的四舍五入位数。
如果要进行四舍五入的操作,则必须依靠以下的方法:
public BigDecimal divide(BigDecimal divisor,int scale,int roundingMode)
scale:表示四舍五入的位数
BigDecimal和RoundingMode是一个绝配,想要采用什么舍入模式使用RoundingMode设置即可。目前Java支持以下七种舍入方式:
ROUND_UP: 远离零方向舍入。
向远离0的方向舍入,也就是说,向绝对值最大的方向舍入,只要舍弃位非0即进位。
ROUND_DOWN:趋向零方向舍入。
向0方向靠拢,也就是说,向绝对值最小的方向输入,注意:所有的位都舍弃,不存在进位情况。
ROUND_CEILING:向正无穷方向舍入。
向正最大方向靠拢,如果是正数,舍入行为类似于ROUND_UP;如果为负数,则舍入行为类似于ROUND_DOWN。注意:Math.round方法使用的即为此模式。
ROUND_FLOOR:向负无穷方向舍入。
向负无穷方向靠拢,如果是正数,则舍入行为类似于 ROUND_DOWN;如果是负数,则舍入行为类似于 ROUND_UP。
HALF_UP: 最近数字舍入(5进)。
这就是我们最最经典的四舍五入模式。
HALF_DOWN:最近数字舍入(5舍)。
在四舍五入中,5是进位的,而在HALF_DOWN中却是舍弃不进位。
HALF_EVEN :银行家算法。
在普通的项目中舍入模式不会有太多影响,可以直接使用Math.round方法,但在大量与货币数字交互的项目中,一定要选择好近似的计算模式,尽量减少因算法不同而造成的损失。(银行家算法就是,假如保留两位,第三位大于5,则进1;小于5,舍去;第三位为奇数,进1,偶数,舍去。)
BigDecimal ba = new BigDecimal(123.015);
System.out.println("计算123.015/1的结果");
System.out.println("ROUND_UP 远离零方向舍入 =" + ba.divide(BigDecimal.ONE, 2,BigDecimal.ROUND_UP));
System.out.println("ROUND_DOWN趋向零方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_DOWN));
System.out.println("ROUND_CEILING向正无穷方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_CEILING));
System.out.println("ROUND_FLOOR向负无穷方向舍入 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_FLOOR));
System.out.println("ROUND_HALF_UP最近数字舍入(5进) =" + ba.divide(BigDecimal.ONE, 2,BigDecimal.ROUND_HALF_UP));
System.out.println("ROUND_HALF_DOWN最近数字舍入(5舍) =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_HALF_DOWN));
System.out.println("ROUND_HALF_EVEN银行家算法 =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_HALF_EVEN));
//System.out.println("ROUND_UNNECESSARY =" + ba.divide(BigDecimal.ONE, 2, BigDecimal.ROUND_UNNECESSARY));
结果
五:测试
@Test
public void testBigDecimal(){
BigDecimal price = BigDecimal.ZERO;
// 通常建议优先使用(String)构造函数
BigDecimal amount = new BigDecimal("6.66");
// 0.06
System.out.println(ArithUtil.add(0.05, 0.01));
// 0.58
System.out.println(ArithUtil.sub(1.0, 0.42));
// 401.5
System.out.println(ArithUtil.mul(4.015, 100));
// 1.233
System.out.println(ArithUtil.div(123.3, 100));
// 4.02
System.out.println(ArithUtil.round(4.015, 2));
// BigDecimal 比较大小使用compareTo方法
// public int compareTo(BigDecimal val);
}
六:注意的问题
- /**
- * (1)BigInteger和BigDecimal都是不可变(immutable)的,在进行每一步运算时,都会产生一个新的对象,由于创建对象会引起开销,
- * 它们不适合于大量的数学计算,应尽量用long,float,double等基本类型做科学计算或者工程计算。
- * 设计BigInteger和BigDecimal的目的是用来精确地表示大整数和小数,使用于在商业计算中使用。
- * (2)BigDecimal有4个够造方法,其中的两个用BigInteger构造,另一个是用double构造,还有一个使用String构造。
- * 应该避免使用double构造BigDecimal,因为:有些数字用double根本无法精确表示,传给BigDecimal构造方法时就已经不精确了。
- * 比如,new BigDecimal(0.1)得到的值是0.1000000000000000055511151231257827021181583404541015625。
- * 使用new BigDecimal("0.1")得到的值是0.1。因此,如果需要精确计算,用String构造BigDecimal,避免用double构造,尽管它看起来更简单!
- * (3)equals()方法认为0.1和0.1是相等的,返回true,而认为0.10和0.1是不等的,结果返回false。
- * 方法compareTo()则认为0.1与0.1相等,0.10与0.1也相等。所以在从数值上比较两个BigDecimal值时,应该使用compareTo()而不是 equals()。
- * (4)另外还有一些情形,任意精度的小数运算仍不能表示精确结果。例如,1除以9会产生无限循环的小数 .111111...。
- * 出于这个原因,在进行除法运算时,BigDecimal可以让您显式地控制舍入。
- */