/*
* 使用BigDecimal需要注意的事项:
* 1、两个BigDecimal值不能使用“ +, -, *, / ” 进行加减乘除,要使用“ add, substract, multiply, divide ”;
* 2、两个BigDecimal值比较使用compareTo方法, 比较结果有-1, 0, 1, 分别表示小于, 等于, 大于; 对于0, 可以使用BigDecimal.ZERO表示;
* 3、提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指 定精度,以后的数字四舍五入;
* 4、因double有效位数为 16位这就会出现存储小数位数不够的情况,这种情况下就会出现误差,解决方法就是使用BigDecimal,它的有
效长度足够长可存储 小数位数因此可代替double来进行加减乘除, 金额必须是完全精确的计算, 故不能使用double或者float, 而
应该采用java.math.BigDecimal.
* 5、mysql数据库设计
BigDecimal在进行入库时, 数据库选择decimal类型, 长度可以自定义, 如18; 小数点我们项目中用的是2, 保留2位小数. 此外还要注意的就是默认值,
一定写成0.00, 不要用默认的NULL, 否则在进行加减排序等操作时, 会带来转换的麻烦!
如 :`balance` decimal(18,2) DEFAULT \'0.00\' COMMENT \'账户余额\',
*/
package bigDecimalTest;
import java.math.BigDecimal;
/**
* 1、两个BigDecimal值不能使用“ +, -, *, / ” 进行加减乘除,要使用“ add, substract, multiply, divide ”
* 2、两个BigDecimal值比较使用compareTo方法, 比较结果有-1, 0, 1, 分别表示小于, 等于, 大于; 对于0, 可以使用BigDecimal.ZERO表示
*
*
*/
public class Arith {
private static final int DEF_DIV_SCALE = 10; // 这个类不能实例化
private Arith() {
}
/**
* 提供精确的加法运算。
*
* @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();
}
/**
* 使用double来进行测试
* @param v1
* @param v2
* @return
*/
public static double sub01(double v1, double v2) {
return v1-v2;
}
/**
* 提供精确的乘法运算。
*
* @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();
}
/**
* 因double有效位数为 16位这就会出现存储小数位数不够的情况,这种情况下就会出现误差,解决方法就是使用BigDecimal,它的有
* 效长度足够长可存储 小数位数因此可代替double来进行加减乘除, 金额必须是完全精确的计算, 故不能使用double或者float, 而
* 应该采用java.math.BigDecimal.
* @param args
*/
public static void main(String[] args) {
//两个BigDecimal值比较使用compareTo方法, 比较结果有-1, 0, 1, 分别表示小于, 等于, 大于;
//对于0, 可以使用BigDecimal.ZERO表示;如:bd.compareTo(BigDecimal.ZERO)
BigDecimal bd = new BigDecimal(3);
BigDecimal bd01 = new BigDecimal(3);
if(bd.compareTo(bd01) > 0){
System.out.println("bd > bd01");
}else if(bd.compareTo(bd01) < 0){
System.out.println("bd < bd01");
}else{
System.out.println("bd = bd01");
}
//减法测试
double s1 = sub(0.03, 0.02);
double s2 = sub01(0.03, 0.02);
System.out.println(s1 + "---" + s2);
// 0.01---0.009999999999999998
//除法测试
System.out.println(div(0.03, 0.022));
}
}
BigDecimal的使用及和浮点型(double)的区别
内容版权声明:除非注明,否则皆为本站原创文章。