dramakevinzz
V2EX  ›  问与答

Java BigDecimal 的四舍五入使用问题

  •  
  •   dramakevinzz · Aug 31, 2022 · 1958 views
    This topic created in 1374 days ago, the information mentioned may be changed or developed.

    又来论坛求助 V 友了,今天在使用 BigDecimal 四舍五入保留一位小数的时候遇到了很神奇的问题。

    double num1 = 5.95;
    double num2 = 6.95;
    double num3 = 7.95;
    double num4 = 8.95;
    double num5 = 9.95;
    double num6 = 5.43;
    System.out.println(new BigDecimal(num1).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 6.0
    System.out.println(new BigDecimal(num2).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 7.0
    System.out.println(new BigDecimal(num3).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 8.0
    System.out.println(new BigDecimal(num4).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 8.9
    System.out.println(new BigDecimal(num5).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 9.9
    System.out.println(new BigDecimal(num6).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 5.4
    

    8.95 和 9.95 四舍五入应该是 9.0 和 10.0 ,但是输出不是这样,但是 5.95 、6.95 、7.95 又不是这样,这是为啥捏

    11 replies    2022-09-02 11:51:06 +08:00
    mango88
        1
    mango88  
       Aug 31, 2022
    System.out.println(new BigDecimal(num1));
    System.out.println(new BigDecimal(num4));
    这俩你输出看一下就明白了,浮点数的精度问题

    另外构建 BigDecimal 对象推荐用字符串构造
    new BigDecimal("5.95")
    new BigDecimal("8.95")
    这样
    dqzcwxb
        2
    dqzcwxb  
       Aug 31, 2022
    mango88
        3
    mango88  
       Aug 31, 2022
    或者 BigDecimal.valueOf(double)
    dqzcwxb
        4
    dqzcwxb  
       Aug 31, 2022
    dramakevinzz
        5
    dramakevinzz  
    OP
       Aug 31, 2022
    @mango88 感谢!这种方法真的就解决了,但是还想请教一下 [浮点数的精度问题] 这个是指什么?可以再解释一下嘛?或者可以再细说一下方向,我查一下,谢谢
    dramakevinzz
        6
    dramakevinzz  
    OP
       Aug 31, 2022
    @dqzcwxb 感谢,这张图确实看着就比较明显了,我记得好像是浮点数的存储和表象并不一致,导致有可能类型 X.9+Y.1 最后并不是 Z.0 这种情况,看 CSAPP 的时候记得这里好像是和 IEEE 设计有关,我再翻一下书,查查,感谢
    mineralsalt
        7
    mineralsalt  
       Aug 31, 2022   ❤️ 1
    因为浮点的问题, 我现在设计金额相关字段的时候, 全部用 int
    dramakevinzz
        8
    dramakevinzz  
    OP
       Aug 31, 2022
    @mineralsalt 金额用 int 吗?这个和日常使用场景怎么契合呢?比如小数,分小数和整数分开存储吗?
    wu67
        9
    wu67  
       Sep 1, 2022   ❤️ 1
    @dramakevinzz 用分为单位, 就不存在小数了
    VeryZero
        10
    VeryZero  
       Sep 1, 2022
    BigDecimal 源码注释里有解释,建议使用字符串构造
    mineralsalt
        11
    mineralsalt  
       Sep 2, 2022
    @dramakevinzz #8 所有金额都用分进行数据库存储, 前端展示的时候格式化一下就行了, 只要存到数据库, 就是分为单位
    About   ·   Help   ·   Advertise   ·   Blog   ·   API   ·   FAQ   ·   Solana   ·   2847 Online   Highest 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 55ms · UTC 01:47 · PVG 09:47 · LAX 18:47 · JFK 21:47
    ♥ Do have faith in what you're doing.