BigDecimal计算
货币金额的计算 - Java中的BigDecimal
在《Effective Java》这本书中也提到这个原则,float和double只能用来做科学计算或者是工程计算,在商业计算中我们要用 java.math.BigDecimal。,而且使用BigDecimal类也可以进行大数的操作。
表11-15 BigDecimal类的常用方法,具体参考API
http://download.java.net/jdk/jdk-api-localizations/jdk-api-zh-cn/builds/latest/html/zh_CN/api/
|
序号 |
方 法 |
类型 |
描 述 |
|
1 |
public BigDecimal(double val) |
构造 |
将double表示形式转换 为BigDecimal |
|
2 |
public BigDecimal(int val) |
构造 |
将int表示形式转换为 BigDecimal |
|
3 |
public BigDecimal(String val) |
构造 |
将字符串表示 形式转换为BigDecimal |
|
4 |
public BigDecimal add(BigDecimal augend) |
普通 |
加法 |
|
5 |
public BigDecimal subtract(BigDecimal |
普通 |
减法 |
|
6 |
public BigDecimal multiply(BigDecimal |
普通 |
乘法 |
|
7 |
public BigDecimal divide(BigDecimal |
普通 |
除法 |
一、 BigDecimal的计算

金额的计算BigDecimal类 double d = 9.84;
double d2 = 1.22;
//注意需要使用BigDecimal(String val)构造方法
BigDecimal bigDecimal = new BigDecimal(Double.toString(d));
BigDecimal bigDecimal2 = new BigDecimal(Double.toString(d2)); //加法
BigDecimal bigDecimalAdd = bigDecimal.add(bigDecimal2);
double add = bigDecimalAdd.doubleValue(); //减法
BigDecimal bigDecimalSubtract = bigDecimal.subtract(bigDecimal2);
double subtract = bigDecimalSubtract.doubleValue(); //乘法
BigDecimal bigDecimalMultiply = bigDecimal.multiply(bigDecimal2);
double multiply = bigDecimalMultiply.doubleValue(); //除法
int scale = 2;//保留2位小数
BigDecimal bigDecimalDivide = bigDecimal.divide(bigDecimal2, scale, BigDecimal.ROUND_HALF_UP);
double divide = bigDecimalDivide.doubleValue(); //格式化
double format = 12343171.6; //获取常规数值格式
NumberFormat number = NumberFormat.getNumberInstance();
String str = number.format(format);//12,343,171.6 //获取整数数值格式
NumberFormat integer = NumberFormat.getIntegerInstance();
str = integer.format(format);//如果带小数会四舍五入到整数12,343,172 //获取货币数值格式
NumberFormat currency = NumberFormat.getCurrencyInstance();
currency.setMinimumFractionDigits(2);//设置数的小数部分所允许的最小位数(如果不足后面补0)
currency.setMaximumFractionDigits(4);//设置数的小数部分所允许的最大位数(如果超过会四舍五入)
str = currency.format(format);//¥12,343,171.60 //获取显示百分比的格式
NumberFormat percent = NumberFormat.getPercentInstance();
percent.setMinimumFractionDigits(2);//设置数的小数部分所允许的最小位数(如果不足后面补0)
percent.setMaximumFractionDigits(3);//设置数的小数部分所允许的最大位数(如果超过会四舍五入)
str = percent.format(format);//1,234,317,160.00%

二、典型的Double类型的数值运算

1 package com.wetalk.wbs.bas.util;
2
3 import java.io.Serializable;
4 import java.math.BigDecimal;
5 import java.math.RoundingMode;
6
7 /**
8 * double的计算不精确,会有类似0.0000000000000002的误差,正确的方法是使用BigDecimal或者用整型
9 * 整型地方法适合于货币精度已知的情况,比如12.11+1.10转成1211+110计算,最后再/100即可
10 * 以下是摘抄的BigDecimal方法:
11 */
12 public class DoubleUtil implements Serializable {
13 private static final long serialVersionUID = -3345205828566485102L;
14 // 默认除法运算精度
15 private static final Integer DEF_DIV_SCALE = 2;
16
17 /**
18 * 提供精确的加法运算。
19 *
20 * @param value1 被加数
21 * @param value2 加数
22 * @return 两个参数的和
23 */
24 public static Double add(Double value1, Double value2) {
25 BigDecimal b1 = new BigDecimal(Double.toString(value1));
26 BigDecimal b2 = new BigDecimal(Double.toString(value2));
27 return b1.add(b2).doubleValue();
28 }
29
30 /**
31 * 提供精确的减法运算。
32 *
33 * @param value1 被减数
34 * @param value2 减数
35 * @return 两个参数的差
36 */
37 public static double sub(Double value1, Double value2) {
38 BigDecimal b1 = new BigDecimal(Double.toString(value1));
39 BigDecimal b2 = new BigDecimal(Double.toString(value2));
40 return b1.subtract(b2).doubleValue();
41 }
42
43 /**
44 * 提供精确的乘法运算。
45 *
46 * @param value1 被乘数
47 * @param value2 乘数
48 * @return 两个参数的积
49 */
50 public static Double mul(Double value1, Double value2) {
51 BigDecimal b1 = new BigDecimal(Double.toString(value1));
52 BigDecimal b2 = new BigDecimal(Double.toString(value2));
53 return b1.multiply(b2).doubleValue();
54 }
55
56 /**
57 * 提供(相对)精确的除法运算,当发生除不尽的情况时, 精确到小数点以后10位,以后的数字四舍五入。
58 *
59 * @param dividend 被除数
60 * @param divisor 除数
61 * @return 两个参数的商
62 */
63 public static Double divide(Double dividend, Double divisor) {
64 return divide(dividend, divisor, DEF_DIV_SCALE);
65 }
66
67 /**
68 * 提供(相对)精确的除法运算。 当发生除不尽的情况时,由scale参数指定精度,以后的数字四舍五入。
69 *
70 * @param dividend 被除数
71 * @param divisor 除数
72 * @param scale 表示表示需要精确到小数点以后几位。
73 * @return 两个参数的商
74 */
75 public static Double divide(Double dividend, Double divisor, Integer scale) {
76 if (scale < 0) {
77 throw new IllegalArgumentException("The scale must be a positive integer or zero");
78 }
79 BigDecimal b1 = new BigDecimal(Double.toString(dividend));
80 BigDecimal b2 = new BigDecimal(Double.toString(divisor));
81 return b1.divide(b2, scale,RoundingMode.HALF_UP).doubleValue();
82 }
83
84 /**
85 * 提供指定数值的(精确)小数位四舍五入处理。
86 *
87 * @param value 需要四舍五入的数字
88 * @param scale 小数点后保留几位
89 * @return 四舍五入后的结果
90 */
91 public static double round(double value,int scale){
92 if(scale<0){
93 throw new IllegalArgumentException("The scale must be a positive integer or zero");
94 }
95 BigDecimal b = new BigDecimal(Double.toString(value));
96 BigDecimal one = new BigDecimal("1");
97 return b.divide(one,scale, RoundingMode.HALF_UP).doubleValue();
98 }
99 }

三、下面提一下两个精度问题:
System.out.println(new BigDecimal(0.1).toString()); // 0.1000000000000000055511151231257827021181583404541015625
System.out.println(new BigDecimal("0.1").toString()); // 0.1
System.out.println(new BigDecimal(
Double.toString(0.1000000000000000055511151231257827021181583404541015625)).toString());// 0.1
System.out.println(new BigDecimal(Double.toString(0.1)).toString()); // 0.1
分析一下上面代码的问题(注释的内容表示此语句的输出)
int x=(int)1023.99999999999999; // x=1024为什么?
原因还是在于二进制无法精确地表示某些十进制小数,因此1023.99999999999999在编译之后的double值变成了1024。
double d = 1023.99999999999999;
int x = (int) d;
System.out.println(new BigDecimal(d).toString()); // 1024
System.out.println(Long.toHexString(Double.doubleToRawLongBits(d))); // 4090000000000000
System.out.println(x); // 1024
前面提过BigDecimal可以精确地把double表示出来还记得吧。
BigDecimal计算的更多相关文章
- Java使用BigDecimal计算保留位数不对问题定位
在项目中使用BigDecimal计算将单位从B换成TB时,算出来的结果总是整数,而没有保留设置的2位小数: 计算代码如下: db.divide(oneTB, ROUND_HALF_UP).setSca ...
- 简化bigdecimal计算的小工具类
简化bigdecimal计算的小工具类 如果我们要做一个加法运算,需要先将两个浮点数转为String,然后够造成BigDecimal,在其中一个上调用add方法,传入另一个作为参数,然后把运算的结果( ...
- Java基础知识强化06:使用BigDecimal计算阶乘1+1/2!+1/3!+……
package himi.hebao04; import java.math.BigDecimal; public class TestDemo07 { public static void main ...
- java 用BigDecimal计算商品单价乘以折扣价
商品单价价格是单位是(分),用户下单金额=商品单价*折扣 代码如下 Integer discount = 5 折扣五折 Integer orderPrice = 1000 单位分 BigDecim ...
- BigDecimal 计算注意事项
BigDecimal 在进行除法运算(divide)时一定要注意:如果被除数为变量,一定要指定精度 和 舍入模式,否则会报:Non-terminating decimal expansion; no ...
- Java浮点数float,bigdecimal和double精确计算的精度误差问题总结
(转)Java浮点数float,bigdecimal和double精确计算的精度误差问题总结 1.float整数计算误差 案例:会员积分字段采用float类型,导致计算会员积分时,7位整数的数据计算结 ...
- 项目中BigDecimal与Double使用场景
金额要用BigDecimal 金额计算不能用doube!!!! 金额计算必须用BigDecimal,下面对比一下用double 跟BigDecimal的区别.先看一个小例子: 请看题: 示例1 问, ...
- 记字符串转bigDecimal的一个坑
项目中一个地方用到了bigdecimal,之前是字符串转Double,处理之后再转成String,看着麻烦,给改成用bigdecimal计算字符串,但是偶尔会出现如下异常. 很是诧异,加了非空校验,怎 ...
- MySQL中Decimal类型和Float Double的区别 & BigDecimal与Double使用场景
MySQL中存在float,double等非标准数据类型,也有decimal这种标准数据类型. 其区别在于,float,double等非标准类型,在DB中保存的是近似值,而Decimal则以字符串的形 ...
随机推荐
- git config 介绍
转载. https://blog.csdn.net/liuxiao723846/article/details/83113317 Git的三个重要配置文件分别是/etc/gitconfig,${HOM ...
- SSM-CRUD
一.项目介绍 前端技术:query+Bootstrap+ajax+json 后端技术:SSM(spring.springMVC.mybatis).JSR303校验 数据库:mysql 服务器:tomc ...
- Java基础笔试练习(三)
1.下列InputStream类中哪个方法可以用于关闭流? A.skip() B.close() C.mark() D.reset() 答案: B 解析: inputstream的close方法用来关 ...
- Python29之字符str与字节bytes
详解见这位大神:https://www.cnblogs.com/xiaobingqianrui/p/9870480.html 实际上字符串和字节之间的转换过程,就是编码解码的过程,我们必须显示的指定编 ...
- python 之 面向对象基础(继承与派生,经典类与新式类)
7.2 继承与派生 7.21继承 1.什么是继承? 继承是一种新建类的的方式,在python中支持一个子类继承多个父类.新建的类称为子类或者派生类,父类又可以称为基类或者超类,子类会”遗传“父类的属性 ...
- Volatile的应用场景
1.当一个变量可能会被意想不到的更新时,要使用volatile来声明该变量,告诉编译器它所修饰的变量的值可能会在任何时刻被意外的更新. 2.语法 volatile int foo; int volat ...
- IDEA使用技巧--将本地项目和git远程项目关联
之前开发没有从头儿搭建过新项目,都是从IDEA配置下项目的git地址,pull代码之后进行开发,提交.这次需要将本地新建的项目push到在git上同样是新建的空项目上去(git上的项目只有工程名和re ...
- SQL根据指定节点ID获取所有父级节点和子级节点(转载)
--根据指定节点ID获取所有子节点-- WITH TEMP AS ( ' --表的主键ID UNION ALL SELECT T0.* FROM TEMP,table_name T0 WHERE TE ...
- 将网址url中的参数转化为JSON格式的两种方法
在我们进入主题前,我先先看下获取网址URL的方法: window.location.href // 设置或获取整个URL为字符串 window.location.hash // 设置或获取href属性 ...
- Microsoft Word (2016) Deceptive File Reference ZDI-CAN-7949
[+] Credits: John Page (aka hyp3rlinx) [+] Website: hyp3rlinx.altervista.org[+] Source: http://hyp3 ...