首页 > 编程语言 >Java正确比较浮点数的方法

Java正确比较浮点数的方法

时间:2022-08-30 11:01:18浏览次数:112  
标签:Java 正确 double 浮点数 sdiff d1 d2 BigDecimal

https://www.jb51.net/article/199220.htm

 

看下面这段代码,将 d1 和 d2 两个浮点数进行比较,输出的结果会是什么?

1 2 3 double d1 = .1 * 3; double d2 = .3; System.out.println(d1 == d2);

按照正常逻辑来看,d1经过计算之后的结果应该是0.3,最后打印的结果应该是 true,对吧?但是运行一下就会发现结果并不是 true 而是 false 

输出一下 d1,发现得到的答案不是想象中的 0.3 而是 0.30000000000000004,所以和 d2 进行比较结果自然是 false

如何正确地比较浮点数(单精度的 float 和双精度的 double),不单单是 Java 特定的问题,在计算机的内存中,存储浮点数时使用的是 IEEE 754 标准,就会有精度的问题。

存储和转换的过程中浮点数容易引起一些较小的舍入误差,正是这个原因,导致在比较浮点数的时候,不能使用“==”操作符——要求严格意义上的完全相等。

那么如何正确的比较浮点数呢?这里有两种方案。

  • 第一种方案是允许两个值之间存在一点误差(指定一个阈值),使用 Math.abs() 方法来计算两个浮点数之间差异的绝对值,如果这个差异在阈值范围之内,我们就认为两个浮点数是相等的。
1 2 3 4 5 6 7 8 9 10 final double THRESHOLD = .0001;   double d1 = .1 * 3; double d2 = .3;   if(Math.abs(d1-d2) < THRESHOLD) {     System.out.println("d1 和 d2 相等"); } else {     System.out.println("d1 和 d2 不相等"); }

Math.abs() 方法用来返回 double 的绝对值,如果 double 小于 0,则返回 double 的正值,否则返回 double。也就是说,abs() 后的结果绝对大于 0,如果结果小于阈值(THRESHOLD),我们就认为 d1 和 d2 相等。

  • 第二种方案是使用 BigDecimal 类,可以指定要舍入的模式和精度,这样就可以解决舍入的误差。

以使用 BigDecimal 类的 compareTo() 方法对两个数进行比较,该方法将会忽略小数点后的位数,怎么理解这句话呢?比如说 2.0 和 2.00 的位数不同,但它俩的值是相等的。

a.compareTo(b) 如果 a 和 b 相等,则返回 0,否则返回 -1。

 

tips: 不要使用 equals() 方法对两个 BigDecimal 对象进行比较,这是因为 equals() 方法会考虑位数,如果位数不同,则会返回 false,尽管数学值是相等的。

1 2 3 4 5 BigDecimal a = new BigDecimal("2.00"); BigDecimal b = new BigDecimal("2.0");   System.out.println(a.equals(b)); System.out.println(a.compareTo(b) == 0);

上面的代码中 a.equals(b) 的结果就为 false,因为 2.00 和 2.0 小数点后的位数不同,但 a.compareTo(b) == 0 的结果就为 true,因为 2.00 和 2.0 在数学层面的值的确是相等的。

compareTo() 方法比较的过程非常严谨,源码如下:

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 private int compareMagnitude(BigDecimal val) {   // Match scales, avoid unnecessary inflation   long ys = val.intCompact;   long xs = this.intCompact;   if (xs == 0)     return (ys == 0) ? 0 : -1;   if (ys == 0)     return 1;     long sdiff = (long)this.scale - val.scale;   if (sdiff != 0) {     // Avoid matching scales if the (adjusted) exponents differ     long xae = (long)this.precision() - this.scale;  // [-1]     long yae = (long)val.precision() - val.scale;   // [-1]     if (xae < yae)       return -1;     if (xae > yae)       return 1;     if (sdiff < 0) {       // The cases sdiff <= Integer.MIN_VALUE intentionally fall through.       if ( sdiff > Integer.MIN_VALUE &&           (xs == INFLATED ||               (xs = longMultiplyPowerTen(xs, (int)-sdiff)) == INFLATED) &&           ys == INFLATED) {         BigInteger rb = bigMultiplyPowerTen((int)-sdiff);         return rb.compareMagnitude(val.intVal);       }     } else { // sdiff > 0       // The cases sdiff > Integer.MAX_VALUE intentionally fall through.       if ( sdiff <= Integer.MAX_VALUE &&           (ys == INFLATED ||               (ys = longMultiplyPowerTen(ys, (int)sdiff)) == INFLATED) &&           xs == INFLATED) {         BigInteger rb = val.bigMultiplyPowerTen((int)sdiff);         return this.intVal.compareMagnitude(rb);       }     }   }   if (xs != INFLATED)     return (ys != INFLATED) ? longCompareMagnitude(xs, ys) : -1;   else if (ys != INFLATED)     return 1;   else     return this.intVal.compareMagnitude(val.intVal); }

接下来,用 BigDecimal 来解决开头的问题。

1 2 3 4 5 6 7 8 9 BigDecimal d1 = new BigDecimal("0.1"); BigDecimal three = new BigDecimal("3"); BigDecimal d2 = new BigDecimal("0.3");   d1 = d1.multiply(three);   System.out.println("d1 = " + d1); System.out.println("d2 = " + d2); System.out.println(d1.compareTo(d2));

程序输出的结果如下:

1 2 3 d1 = 0.3 d2 = 0.3 0

d1 和 d2 都为 0.3,所以 compareTo() 的结果就为 0,表示两个值是相等的。

总结一下,在遇到浮点数的时候,千万不要使用 == 操作符来进行比较,因为有精度问题。要么使用阈值来忽略舍入的问题,要么使用 BigDecimal 来替代 double 或者 float。

以上就是Java正确比较浮点数的方法的详细内容,更多关于Java 正确比较浮点数的资料请关注脚本之家其它相关文章!

 

 

 

 

标签:Java,正确,double,浮点数,sdiff,d1,d2,BigDecimal
From: https://www.cnblogs.com/kelelipeng/p/16638565.html

相关文章

  • 降低Java垃圾回收开销的5条建议
    保持GC低开销的窍门有哪些?随着一再拖延而即将发布的Java9,G1(“GarbageFirst”)垃圾回收器将被成为HotSpot虚拟机默认的垃圾回收器。从serial垃圾回收器到C......
  • java枚举
    ......
  • Java中Thread类的常用API以及使用示例
    场景Java语言是支持多线程的,一个正在运行的Java程序可以称之为一个进程(process),在每个进程里面包含多个线程,线程是进程中单一的顺序控制流,CPU在执行计算机指令的时候都是......
  • B/S端界面控件DevExtreme JavaScript—全新的UI模板库 (v22.2)
    DevExtreme拥有高性能的HTML5/JavaScript小部件集合,使您可以利用现代Web开发堆栈(包括React,Angular,ASP.NETCore,jQuery,Knockout等)构建交互式的Web应用程序,该套件附带功能......
  • nacos2.1 新增配置发布失败。请检查参数是否正确
    使用官方的docker部署方式,部署了一个单节点nacosserver,部署完了后发布配置信息,报错“新增配置发布失败。请检查参数是否正确”解决方法:在nacosmysql数据库中分别在conf......
  • java计算代码段执行时间
    java里计算代码段执行时间可以有两种方法,一种是毫秒级别的计算,另一种是更精确的纳秒级别的计算。一)毫秒级别计算时间longstartTime=System.currentTimeMillis();/*......
  • Java 学生管理系统
    今天简单练习了一下数组和循环的一些知识,写了一个小游戏,大致如下/***一个小功能:学生管理系统*1、添加学生信息*2、显示学生信息*3、删除学生信息*4、修......
  • 你对 JavaScript 中的变量了解多少?
    你知道多少JavaScript中的变量?昨天,当我搞砸了变量时,我正在研究我的开尔文天气项目。该项目需要声明许多变量以将温度程度从一个更改为另一个。因此对于像我这样的Roo......
  • Javascript 中的内存引用
    Javascript中的内存引用Photoby哈里森布罗德本特on不飞溅在本文中,我将尝试通过一个示例练习来解释Javascript中的内存引用是如何处理的,我认为这可以更好地展示......
  • JAVA SE
    注释//adf单行​/*aadf多行adsfadf*/ 关键字&标识符关键字像voidmainpublic这种固定的单词有特殊含义的编写时会变色不能用作类名,变量名方法名标识符......