Double与BigDecimal 比较
1] 精确的浮点运算:
在Java里面,有时候为了保证数值的准确性需要精确的数据,先提供一个例子就可以发现问题了:
- public class FloatNumberTester {
- public static void main(String args[]){
- System.out.println(0.05+0.01);
- System.out.println(1.0 - 0.42);
- System.out.println(4.015 * 100);
- System.out.println(123.3 / 100);
- }
- }
按照我们的期待,上边应该是什么结果呢,但是看输出我们就会发现问题了:
- 0.060000000000000005
- 0.5800000000000001
- 401.49999999999994
- 1.2329999999999999
这样的话这个问题就相对严重了,如果我们使用123.3元交易,计算机却因为1.2329999999999999而拒绝了交易,岂不是和实际情况大相径庭。
[2] 四舍五入:
另外的一个计算问题,就是四舍五入。但是Java的计算本身是不能够支持四舍五入的,比如:
- public class GetThrowTester {
- public static void main(String args[]){
- System.out.println(4.015 * 100.0);
- }
- }
这个输出为:
401.49999999999994
所以就会发现这种情况并不能保证四舍五入,如果要四舍五入,只有一种方法
java.text.DecimalFormat:
- import java.text.DecimalFormat;
- public class NumberFormatMain {
- public static void main(String args[]){
- System.out.println(new DecimalFormat("0.00").format(4.025));
- System.out.println(new DecimalFormat("0.00").format(4.024));
- }
- }
上边代码输出为:
- 4.02
- 4.02
发现问题了么?因为DecimalFormat使用的舍入模式, 舍入模式 详情参见本文最后部分。
[3] 浮点输出:
Java浮点类型数值在大于9999999.0就自动转化成为科学计数法,看看下边的例子:
- public class FloatCounter {
- public static void main(String args[]){
- System.out.println(9969999999.04);
- System.out.println(199999999.04);
- System.out.println(1000000011.01);
- System.out.println(9999999.04);
- }
- }
输出结果为:
- 9.96999999904E9
- 1.9999999904E8
- 1.00000001101E9
- 9999999.04
但是有时候我们不需要科学计数法,而是转换成为字符串,所以这样可能会有点麻烦。
总结:
所以在项目当中,对于浮点类型以及大整数的运算 还是尽量不要用double,long等基本数据类型以及其包装类,还是用Java中提供的BigDecimal,BigInteger等大数值类型来代替吧。
但这里特别说明一下BigDecimal类的两个构造函数的区别,他们分别是:
new BigDecimal(String val ) 和 new BigDecimal(double val )
先看例子:
- public class BigDecimalMain {
- public static void main(String args[]){
- System.out.println(new BigDecimal(123456789.01).toString());
- System.out.println(new BigDecimal("123456789.01").toString());
- }
- }
输出结果有一次令人意外了,同时两者之间的区别也一目了然了:
- 123456789.01000000536441802978515625
- 123456789.01
所以在 就是想利用double原始类型进行了相关计算之后再转成BigDecimal类型 的场合下,为了防止精度出现偏离,建议使用参数为String类型的该构造方法。即new BigDecimal(String val )。
BigDecimal舍入模式介绍:
舍入模式在java.math.RoundingMode 里面:
RoundingMode.CEILING :向正无限大方向舍入的舍入模式。如果结果为正,则舍入行为类似于 RoundingMode.UP;如果结果为负,则舍入行为类似于 RoundingMode.DOWN。注意,此舍入模式始终不会减少计算值
| 输入数字 | 使用CEILING舍入模式将数字舍入为一位数 |
| 5.5 | 6 |
| 2.5 | 3 |
| 1.1 | 2 |
| 1.0 | 1 |
| -1.0 | -1 |
| -1.1 | -1 |
| -1.6 | -1 |
| -2.5 | -2 |
| -5.5 | -5 |
RoundingMode.DOWN :向零方向舍入的舍入模式。从不对舍弃部分前面的数字加 1(即截尾)。注意,此舍入模式始终不会增加计算值的绝对值
| 输入数字 | 使用DOWN舍入模式将数字舍入为一位数 |
| 5.5 | 5 |
| 2.5 | 2 |
| 1.1 | 1 |
| -1.0 | -1 |
| -1.6 | -1 |
| -2.5 | -2 |
| -5.5 | -5 |
RoundingMode.FLOOR :向负无限大方向舍入的舍入模式。如果结果为正,则舍入行为类似于 RoundingMode.DOWN;如果结果为负,则舍入行为类似于 RoundingMode.UP。注意,此舍入模式始终不会增加计算值
| 输入数字 | 使用FLOOR舍入模式将输入数字舍入为一位 |
| 5.5 | 5 |
| 2.3 | 2 |
| 1.6 | 1 |
| 1.0 | 1 |
| -1.1 | -2 |
| -2.5 | -3 |
| -5.5 | -6 |
RoundingMode.HALF_DOWN :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向下舍入。如果被舍弃部分 > 0.5,则舍入行为同 RoundingMode.UP;否则舍入行为同 RoundingMode.DOWN
| 输入数字 | 使用HALF_DOWN输入模式舍入为一位 |
| 5.5 | 5 |
| 2.5 | 2 |
| 1.6 | 2 |
| 1.0 | 1 |
| -1.1 | -1 |
| -1.6 | -2 |
| -2.5 | -2 |
| -5.5 | -5 |
RoundingMode.HALF_EVEN :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向相邻的偶数舍入。如果舍弃部分左边的数字为奇数,则舍入行为同 RoundingMode.HALF_UP;如果为偶数,则舍入行为同 RoundingMode.HALF_DOWN。注意,在重复进行一系列计算时,此舍入模式可以在统计上将累加错误减到最小。此舍入模式也称为“银行家舍入法”,主要在美国使用。此舍入模式类似于 Java 中对 float 和 double 算法使用的舍入策略
| 输入数字 | 使用HALF_EVEN舍入模式将输入舍为一位 |
| 5.5 | 6 |
| 2.5 | 2 |
| 1.6 | 2 |
| 1.1 | 1 |
| -1.0 | -1 |
| -1.6 | -2 |
| -2.5 | -2 |
| -5.5 | -6 |
RoundingMode.HALF_UP :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向上舍入。如果被舍弃部分 >= 0.5,则舍入行为同 RoundingMode.UP;否则舍入行为同 RoundingMode.DOWN。注意,此舍入模式就是通常学校里讲的四舍五入
| 输入数字 | 使用HALF_UP舍入模式舍入为一位数 |
| 5.5 | 6 |
| 2.5 | 3 |
| 1.6 | 2 |
| 1.0 | 1 |
| -1.1 | -1 |
| -1.6 | -2 |
| -2.5 | -3 |
| -5.5 | -6 |
RoundingMode.UNNECESSARY :用于断言请求的操作具有精确结果的舍入模式,因此不需要舍入。如果对生成精确结果的操作指定此舍入模式,则抛出 ArithmeticException
| 输入数字 | 使用UNNECESSARY模式 |
| 5.5 | 抛出 ArithmeticException |
| 2.5 | 抛出 ArithmeticException |
| 1.6 | 抛出 ArithmeticException |
| 1.0 | 1 |
| -1.0 | -1.0 |
| -1.1 | 抛出 ArithmeticException |
| -1.6 | 抛出 ArithmeticException |
| -2.5 | 抛出 ArithmeticException |
| -5.5 | 抛出 ArithmeticException |
RoundingMode.UP :远离零方向舍入的舍入模式。始终对非零舍弃部分前面的数字加 1。注意,此舍入模式始终不会减少计算值的绝对值
| 输入数字 | 使用UP舍入模式将输入数字舍入为一位数 |
| 5.5 | 6 |
| 1.6 | 2 |
| 1.1 | 2 |
| 1.0 | 1 |
| -1.1 | -2 |
| -1.6 | -2 |
| -2.5 | -3 |
| -5.4 | -6 |
- import java.math.BigDecimal;
- import java.text.DecimalFormat;
- /**
- *使用舍入模式的格式化操作
- **/
- public class DoubleFormat {
- public static void main(String args[]){
- DoubleFormat format = new DoubleFormat();
- System.out .println(format.doubleOutPut(12.345, 2));
- System.out .println(format.roundNumber(12.335, 2));
- }
- public String doubleOutPut(double v,Integer num){
- if ( v == Double.valueOf(v).intValue()){
- return Double.valueOf(v).intValue() + "" ;
- }else {
- BigDecimal b = new BigDecimal(Double.toString(v));
- return b.setScale(num,BigDecimal.ROUND_HALF_UP ).toString();
- }
- }
- public String roundNumber(double v,int num){
- String fmtString = "0000000000000000" ; //16bit
- fmtString = num>0 ? "0." + fmtString.substring(0,num):"0" ;
- DecimalFormat dFormat = new DecimalFormat(fmtString);
- return dFormat.format(v);
- }
- }
这段代码的输出为:
- 12.35
- 12.34
Double与BigDecimal 比较的更多相关文章
- [转]double与BigDecimal
转自:http://superivan.iteye.com/blog/963628 [1] 精确的浮点运算: 在Java里面,有时候为了保证数值的准确性需要精确的数据,先提供一个例子就可以发现问题了: ...
- float、double、BigDecimal的一些精度问题
float f = 280.8f;System.out.println(f*100);结果是什么?结果是:28080.0f(我是这么想的)实际结果是:28079.998 既然float处理有问题换do ...
- 【1】Java中double转BigDecimal的注意事项
项目遇到该问题 先上结论:不要直接用double变量作为构造BigDecimal的参数. 线上有这么一段Java代码逻辑: 1,接口传来一个JSON串,里面有个数字:57.3. 2,解析JSON并把这 ...
- java中四舍五入——double转BigDecimal的精度损失问题
代码: double d = -123456789012345.3426;//5898895455898954895989; NumberFormat nf = new DecimalFormat(& ...
- Java中double转BigDecimal的注意事项
先上结论:不要直接用double变量作为构造BigDecimal的参数! 线上有这么一段Java代码逻辑: 1,接口传来一个JSON串,里面有个数字:57.3. 2,解析JSON并把这个数字保存在一个 ...
- Double与BigDecimal 精度问题
转自:http://superivan.iteye.com/blog/963628 [1] 精确的浮点运算: 在Java里面,有时候为了保证数值的准确性需要精确的数据,先提供一个例子就可以发现问题了: ...
- 小数数据精度问题Double与BigDecimal
做项目的过程中涉及到小数问题的时候,一般我都用Double类型,但是经常出现*.999999998这种数据,然后自己再手动四舍五入,简直傻的要死. 明明就是一个1.51-1.38的问题,很简单怎么会得 ...
- Char、float、Double、BigDecimal
Char初识 char: char类型是一个单一的 16 位 Unicode 字符 char 在java中是2个字节("字节"是byte,"位"是bit ,1 ...
- Java浮点数float,bigdecimal和double精确计算的精度误差问题总结
(转)Java浮点数float,bigdecimal和double精确计算的精度误差问题总结 1.float整数计算误差 案例:会员积分字段采用float类型,导致计算会员积分时,7位整数的数据计算结 ...
随机推荐
- CI 笔记5 (CI3.0 默认控制器,多目录)
在ci3.x中,不支持多级子目录的默认控制器设置, 解决方法如下: 在index.php中,添加 $routing['directory'] = 'admin';然后在默认的router.php的默 ...
- linux常用命令之ln
ln是linux中又一个非常重要命令,它的功能是为某一个文件在另外一个位置建立一个同不的链接,这个命令最常用的参数是-s,具体用法是:ln –s 源文件 目标文件. 当我们需要在不同的目录,用到相同的 ...
- 常用的dos命令之简略总结
Dos常用命令 一.基础命令 1 dir 无参数:查看当前所在目录的文件和文件夹. /s:查看当前目录已经其所有子目录的文件和文件夹. /a:查看包括隐含文件的所有文件. /ah:只显示出 ...
- JavaScript HTML DOM 元素(节点)
JavaScript HTML DOM 元素(节点) 创建新的 HTML 元素 创建新的 HTML 元素 如需向 HTML DOM 添加新元素,您必须首先创建该元素(元素节点),然后向一个已存在的元素 ...
- 【转】WF4.0实战系列索引
转自:http://www.cnblogs.com/zhuqil/archive/2010/07/05/wf4-in-action-index.html 此系列的工作流文件案例比较多点,实用性好. W ...
- 【POJ2761】【区间第k大】Feed the dogs(吐槽)
Description Wind loves pretty dogs very much, and she has n pet dogs. So Jiajia has to feed the dogs ...
- Eclipse 编译StanfordNLP
1.源码最新下载地址:http://nlp.stanford.edu/software/index.shtml; 2.解压stanford-corenlp.zip; 3.打开Eclipse新建JAVA ...
- Linux on ASUS N550JK4700
实际上,ASUS N550JK对Ubuntu 14.04的兼容性是相当好的,包括无线网卡.蓝牙.键盘背光的调节.触摸板的开关.音量的键盘调节都是安装后无需配置直接可以使用的,这是出乎意料的,因为这些功 ...
- [Python]更加Pythonic的多个List合并和Python的安利
原题: https://segmentfault.com/q/1010000005904259 问题: 倘若存在 L=[ [1,2,3],[4,5,6],[7,8,9]] 这样的列表,如何把合并成[1 ...
- NameNode元数据的管理机制(三)
元数据的管理: 第一步:客户端通过DistributedFilesystem 对象中的creat()方法来创建文件,此时,RPC会 通过一个RPC链接协议来调用namenode,并在命名空间中创建一个 ...