zoukankan      html  css  js  c++  java
  • JAVA正确的四舍五入方法

    在JDK版本为1.8的情况运行下面的代码,会发现很神奇的情况(见运行结果)。

    看如下代码:

    package com.longge.mytest;
    
    import java.math.BigDecimal;
    import java.math.RoundingMode;
    import java.text.DecimalFormat;
    
    import org.junit.Test;
    
    public class TestDecimal {
        @Test
        public void test() {
            BigDecimal a = new BigDecimal(0.075);
            BigDecimal a1 = new BigDecimal(10.075);
            BigDecimal a2 = new BigDecimal(100.075);
            
            BigDecimal b = new BigDecimal(0.074);
            BigDecimal b1 = new BigDecimal(10.074);
            BigDecimal b2 = new BigDecimal(100.074);
            
            BigDecimal c = new BigDecimal("0.075");
            BigDecimal c1 = new BigDecimal("10.075");
            BigDecimal c2 = new BigDecimal("100.075");
            
            DecimalFormat df = new DecimalFormat("#0.00");
            
            BigDecimal one = BigDecimal.ONE;
            
            // 0.07
            System.out.println(a.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 0.08
            System.out.println(a.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 0.07
            System.out.println(b.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 0.07
            System.out.println(b.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 0.07
            System.out.println(df.format(0.074));
            // 0.07
            System.out.println(df.format(0.075));
            // 0.08
            System.out.println(c.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            
            System.out.println("----------------------------");
            
            // 10.07
            System.out.println(a1.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 10.08
            System.out.println(a1.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 10.7
            System.out.println(b1.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 10.07
            System.out.println(b1.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 10.07
            System.out.println(df.format(10.074));
            // 10.07
            System.out.println(df.format(10.075));
            // 10.08
            System.out.println(c1.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            
            System.out.println("----------------------------");
            
            // 100.08
            System.out.println(a2.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 100.08
            System.out.println(a2.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 100.07
            System.out.println(b2.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
            // 100.07
            System.out.println(b2.divide(one).add(new BigDecimal("0.0000000001")).setScale(2, RoundingMode.HALF_UP).doubleValue());
            // 100.07
            System.out.println(df.format(100.074));
            // 100.08
            System.out.println(df.format(100.075));
            // 100.08
            System.out.println(c2.divide(one, 2, RoundingMode.HALF_UP).doubleValue());
        }
    }

    运行结果如下:

    0.07
    0.08
    0.07
    0.07
    0.07
    0.07
    0.08
    ----------------------------
    10.07
    10.08
    10.07
    10.07
    10.07
    10.07
    10.08
    ----------------------------
    100.08
    100.08
    100.07
    100.07
    100.07
    100.08
    100.08

    其中可选模式有:

    1、 ROUND_UP:远离零方向舍入。向绝对值最大的方向舍入,只要舍弃位非0即进位。

    2、 ROUND_DOWN:趋向零方向舍入。向绝对值最小的方向输入,所有的位都要舍弃,不存在进位情况。

    3、 ROUND_CEILING:向正无穷方向舍入。向正最大方向靠拢。若是正数,舍入行为类似于ROUND_UP,若为负数,舍入行为类似于ROUND_DOWN。Math.round()方法就是使用的此模式。

    4、 ROUND_FLOOR:向负无穷方向舍入。向负无穷方向靠拢。若是正数,舍入行为类似于ROUND_DOWN;若为负数,舍入行为类似于ROUND_UP。

    5、 HALF_UP:最近数字舍入(5进)。这是我们最经典的四舍五入。

     6、 HALF_DOWN:最近数字舍入(5舍)。在这里5是要舍弃的。

    7、 HAIL_EVEN:银行家舍入法。

    查看JDK的官方说明:

    所以在做对精度要求高的计算时,要将double/float转成字符串后再转成BigDecimal进行计算

  • 相关阅读:
    Fire and Motion[转载]
    HLSL2GLSL v0.9 Released
    CEGUI Release of 0.5.0 stable by CrazyEddie 6th November 2006
    MapInfo 连接Oracle
    MapInfo连接SQLServer
    视图的创建及使用(sql server 2005)
    MapInfo 建立永久表
    MapInfo Update Feature
    MapInfo导入.TAB和.mws的方法
    触发器的创建及使用(sqlserver 2000)
  • 原文地址:https://www.cnblogs.com/yangzhilong/p/8405840.html
Copyright © 2011-2022 走看看