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则以字符串的形 ...
随机推荐
- C之typedef应用
1.0关于typedef关键字的基础: https://www.cnblogs.com/anSn/p/8783347.html 1.1 typedef 修饰“函数类型” 的调用方法: 1)我们写一段普 ...
- WUSTOJ 1321: Alphabet Cookies(Java)字符统计
题目链接:1321: Alphabet Cookies Description Kitty likes cookies very much, and especially the alphabet c ...
- netty--处理器
编解码器的基类: 编码:MessageToByteEncode 解码:ByteToMessageDecoder
- Security Access Control Strategy && Method And Technology Research - 安全访问控制策略及其方法技术研究
1. 访问控制基本概念 访问控制是网络安全防范和客户端安全防御的重要基础策略,它的主要任务是保证资源不被非法使用.保证网络/客户端安全最重要的核心策略之一. 访问控制包括 入网访问控制 网络权限控制 ...
- 使用Spring Cloud OAuth2和JWT保护微服务
采用Spring Security AOuth2 和 JWT 的方式,避免每次请求都需要远程调度 Uaa 服务.采用Spring Security OAuth2 和 JWT 的方式,Uaa 服务只验证 ...
- hdu 1045 要求全部逐一搜索完的深搜
#include<stdio.h> #include<string.h> int visit[10][10]; char map[10][10]; int n,ans,ss,t ...
- windows下mysql表名区分大小写
windows下mysql默认是不区分大小写的,但是linux会区分大小写 如何让windows下mysql区分大小写呢? 修改 my.ini 里面的mysqld部分 #区分大小写 lower_cas ...
- 什么叫工业4.0,这篇接地气的文章终于讲懂了(ZT)
原地址:https://www.cnblogs.com/namei/p/6110382.html 笔者早年从事过工业自动化行业,后来去了几个城市,讲过<工业互联网与工业文明史>这门课,以至 ...
- C#不支持XPATH2.0
.net中的XPATH是1.0版本的,很多2.0中的函数是不兼容的,比如lower-case().replace()函数等,下面中的XPATH语句在运行时会报错 //table[contains(lo ...
- k8s 开源web操作平台
https://kuboard.cn/install/install-dashboard.html kuborad