double和float不能进行精确计算,商业计算经常使用BigDecimal进行精确计算。
1. 创建BigDecimal类型数据
使用构造方法:
可将基本类型数据、String类型(可转为基础类型的)、bigInteger等类型通过BigDecimal构造函数方式转为BigDecimal类型数据。
public BigDecimal(String val);
public BigDecimal(double val);
public BigDecimal(int val);
public BigDecimal(long val);
使用BigDecimal.valueOf()方法:
public static BigDecimal valueOf(double val);
将double类型数据转为BigDecimal类型:
使用BigDecimal的double类型构造函数转化,结果不精确:
BigDecimal bg = new BigDecimal(0.1);
System.out.println(bg);
期望输出:0.1
实际输出:0.1000000000000000055511151231257827021181583404541015625
为了得到精确值,double转为BigDecimal类型,使用以下三种方法:
1). 先将double val 转为String类型,再使用BigDecimal的String类型构造函数
BigDecimal bg= new BigDecimal(String.valueOf(val));
2). 使用BigDecimal的double类型valueOf()方式
BigDecimal bg= BigDecimal.valueOf(val);
3).先对double类型数据进行格式化,然后再转化:
DecimalFormat df = new DecimalFormat("###0.00");//保留两位小数
BigDecimal bg= new BigDecimal(df.format(val));
2. BigDecimal常用方法
加法:public BigDecimal add(BigDecimal augend);
减法:public BigDecimal subtract(BigDecimal subtrahend);
乘法:public BigDecimal multiply(BigDecimal multiplicand);
除法:public BigDecimal divide(BigDecimal divisor, int roundingMode);
设置精度:public BigDecimal setScale(int newScale, RoundingMode roundingMode);newScale小数点后面保留位数,roundingMode取舍方式;
比较:public int compareTo(BigDecimal val);
用法示例:
package test;
import java.math.BigDecimal;
import java.text.DecimalFormat;
/**
* @author sunzl
* @date 2018年10月14日
*
*/
public class BigDecimalTest {
public static void main(String[] args) {
DecimalFormat df = new DecimalFormat("###0.00");//保留两位小数
double val1 = 0.11111;
double val2 = 0.99999;
BigDecimal bg1 = new BigDecimal(df.format(val1));
BigDecimal bg2 = new BigDecimal(df.format(val2));
System.out.println(bg1);
System.out.println(bg2);
System.out.println(bg1.add(bg2).doubleValue());
System.out.println(bg1.subtract(bg2).doubleValue());
System.out.println(bg1.multiply(bg2).setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue()); //保留两位小数
System.out.println(bg1.divide(bg2).setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue());
}
}
输出:
0.11
1.00
1.11
-0.89
0.11
0.11
对于乘法和除法,为了得到精确的小数位数,需要设置精度,通过setScale(int newScale, RoundingMode roundingMode)进行设置,newScale为需要保留的小数点位数,roundingMode精确方式,其中,除法如果除不尽,必须设置精度,否则报错。
常用精确方式:
ROUND_HALF_UP 四舍五入
ROUND_CEILING 向正无穷方向舍入
ROUND_DOWN 向零方向舍入
ROUND_FLOOR 向负无穷方向舍入
ROUND_HALF_DOWN 向(距离)最近的一边舍入