zoukankan      html  css  js  c++  java
  • Java比较两个浮点数

    浮点数的基本数据类型不能用==比较,包装数据类型不能用 equals 比较

    浮点数的表示

    在计算机系统中,浮点数采用 符号+阶码+尾数 进行表示。在Java中,单精度浮点数float类型占32位,它的二进制表示方式为:

    • 符号位:1位,0表示正数; 1表示负数
    • 指数位:8位,用来表示指数(要加上偏移量)
    • 小数位:23位,用来表示小数

    实际上计算机中的浮点数表示方法和科学技术法类似,小数的位数决定了浮点数的精度。当一个十进制的小数转换成二进制时,很有可能无法使用这么多小数位表示它。因此使用浮点数的时候,实际存储的尾数是被截取或者舍入之后的值。因此使用浮点数进行计算的时候就不得不考虑精度问题,即浮点数无法精确计算。

    如何比较浮点数

    1. 不能使用==

    对于浮点数,我们无法使用==或者包装类型的equals()来精确比较。例如:

    // 对f1执行11次加0.1的操作
    float f1 = 0.0f;
    for (int i = 0; i < 11; i++) {
        f1 += 0.1f;
    }
    // f2是0.1*11的结果
    float f2 = 0.1f * 11;
    System.out.println(f1 == f2);   // 结果为false
    

    执行上述代码,会得到二者不相等,同时如果打印f1和f2,会得到如下结果:

    f1 = 1.1000001
    f2 = 1.1
    

    可以看到,在浮点数的运算过程中,确实出现了精度问题。

    2. 规定误差范围

    尽管无法做到精确比较,但是我们可以确定一个误差范围,当小于这个误差范围的时候就认为这两个浮点数相等。例如:

    final float THRESHOLD = 0.000001;   // 设置最大误差不超过0.000001
    float f1 = 0.0f;
    for (int i = 0; i < 11; i++) {
        f1 += 0.1f;
    }
    
    float f2 = 0.1f * 11;
    
    if (Math.abs(f1 - f2) < THRESHOLD) {
        System.out.println("f1 equals f2");
    }
    

    3. 使用BigDecimal

    BigDecimal是一个不可变的、能够表示大的十进制整数的对象。注意使用BigDecimal时,要使用参数为String的构造方法,而不要使用参数为double的构造方法,防止产生精度丢失。使用BigDecimal进行运算,使用它的compareTo()方法比较即可。
    示例:

    private void compareByBigDecimal() {
        BigDecimal f1 = new BigDecimal("0.0");
        BigDecimal pointOne = new BigDecimal("0.1");
        for (int i = 0; i < 11; i++) {
            f1 = f1.add(pointOne);
        }
    
        BigDecimal f2 = new BigDecimal("0.1");
        BigDecimal eleven = new BigDecimal("11");
        f2 = f2.multiply(eleven);
    
        System.out.println("f1 = " + f1);
        System.out.println("f2 = " + f2);
    
        if (f1.compareTo(f2) == 0) {
            System.out.println("f1 and f2 are equal using BigDecimal");
        } else {
            System.out.println("f1 and f2 are not equal using BigDecimal");
        }
    }
    

    参考链接

  • 相关阅读:
    Freemaker:操作集合
    win10:在关闭防火墙下如何屏蔽特定端口
    win10:家庭版开启组策略
    oracle:10g下载地址(转载)
    mybatis:延迟加载时不要在get/set方法上面添加final关键字(原创)
    mybatis:在springboot中的配置
    mybatis:访问静态变量或方法
    maven:手动安装JAR到本地仓库
    datatables日常使用集合
    python 装饰器
  • 原文地址:https://www.cnblogs.com/zhloong/p/java-float-number-compare.html
Copyright © 2011-2022 走看看