Java浮点型数值的精确运算——BigDecimal
在一个项目中碰到了这样的问题:在进行复杂浮点运算的时候,往往精度保存的不准确。比如在我之前的一个电商项目上,购物车总价目测结果应该是42.11,结果得到街结果却是42.1099999999999。之后想起了Java浮点类型数据使用的是二进制科学计数法表示的,不能精确表示。我已经使用了double双精度浮点数,还是偶尔会出现这样的问题。
不过Java提供了一个BigDecimal
类来精确地表示一个浮点类型数,具体用法可以通过JavaAPI文档查询。
下面贴出解决浮点数精确运算的工具类。
public class ArithUtil {
/**
* 对double数据进行取精度.
*
* @param value double数据.
* @param scale 精度位数(保留的小数位数).
* @param roundingMode 精度取值方式.
* @return 精度计算后的数据.
*/
public static double round(double value, int scale,
int roundingMode) {
BigDecimal bd = new BigDecimal(value);
bd = bd.setScale(scale, roundingMode);
double d = bd.doubleValue();
bd = null;
return d;
}
/**
* double 相加
*
* @param d1
* @param d2
* @return
*/
public static double sum(double d1, double d2) {
BigDecimal bd1 = new BigDecimal(Double.toString(d1));
BigDecimal bd2 = new BigDecimal(Double.toString(d2));
return bd1.add(bd2).doubleValue();
}
/**
* double 相减
*
* @param d1
* @param d2
* @return
*/
public static double sub(double d1, double d2) {
BigDecimal bd1 = new BigDecimal(Double.toString(d1));
BigDecimal bd2 = new BigDecimal(Double.toString(d2));
return bd1.subtract(bd2).doubleValue();
}
/**
* double 乘法
*
* @param d1
* @param d2
* @return
*/
public static double mul(double d1, double d2) {
BigDecimal bd1 = new BigDecimal(Double.toString(d1));
BigDecimal bd2 = new BigDecimal(Double.toString(d2));
return bd1.multiply(bd2).doubleValue();
}
/**
* double 除法
*
* @param d1
* @param d2
* @param scale 四舍五入 小数点位数
* @return
*/
public static double div(double d1, double d2, int scale) {
// 当然在此之前,你要判断分母是否为0,
// 为0你可以根据实际需求做相应的处理
BigDecimal bd1 = new BigDecimal(Double.toString(d1));
BigDecimal bd2 = new BigDecimal(Double.toString(d2));
return bd1.divide
(bd2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
}
}