zoukankan      html  css  js  c++  java
  • 【踩坑系列】使用long类型处理金额,科学计数法导致金额转大写异常

    1. 踩坑经历

    上周,一个用户反馈他创建的某个销售单无法打开,但其余销售单都可以正常打开,当时查看了生产环境的ERROR日志,发现抛了这样的异常:java.lang.NumberFormatException: For input string: "E"

    相信大家对这个异常都不陌生,很显然,是因为将字符串转换为数字时抛出的,比如下面这样:

    但仔细查看了用户报错的单据,也没有发现哪里有输入“E”这样的字符串(请原谅我第一时间没有想到是科学计数法造成的,哈哈),最后把生产环境的这条数据插入到了开发环境中,定位到原来是因为将金额转换为大写时导致的,报错的关键代码如下所示:

    String totalAmountStr = String.valueOf(totalAmount / 100.0);
    
    String amountCN = MoneyUtils.toChinese(totalAmountStr);
    

    其中totalAmount是一个long类型的变量,之所以除以100.0,是因为我们数据库中存储金额都是按为单位存储的(相信很多小伙伴也是这么存储的),第2行代码主要是为了将金额转换为大写,比如将105000.50转换为壹拾万零伍仟元伍角。

    用户报错的那个单据,totalAmount为2700万,转换为分就是:2700000000,执行完totalAmount / 100.0,输出结果竟然是2.7E7,而不是预期的27000000,因此导致了异常:java.lang.NumberFormatException: For input string: "E"

    最后的解决方案是将金额转换为BigDecimal来处理,代码修改为如下所示:

    String totalAmountStr = new BigDecimal(String.valueOf(totalAmount)).divide(new BigDecimal("100"),2, RoundingMode.HALF_UP).toString();
    String amountCN = MoneyUtils.toChinese(totalAmountStr);
    

    2. 原因分析

    在Java中,当浮点数(float、double)的整数部分达到8位及以上,会以科学计数法表示,如下所示:

    double firstAmount = 2700000D;
    double secondAmount = 27000000D;
    double thirdAmount = 2700000.25D;
    double fourthAmount = 27000000.25D;
    
    System.out.println(firstAmount);
    System.out.println(secondAmount);
    System.out.println(thirdAmount);
    System.out.println(fourthAmount);
    

    默默数了下,整数部分8位的话,都是千万级别了,估计遇到这个问题的用户很豪,哈哈。

    所以使用double来表示金额,当金额遇到科学计数法时,就会显示不正常、甚至造成一些意想不到的异常。

    3. 解决方案

    如果不想用科学计数法显示,而是显示金额本身,有以下2种解决方案:

    1. 使用NumberFormat
    2. 使用BigDecimal

    3.1 方案一:使用NumberFormat

    使用NumberFormat的方法如下所示:

    NumberFormat numberFormat = NumberFormat.getInstance();
    numberFormat.setGroupingUsed(false);
    
    double secondAmount = 27000000D;
    double fourthAmount = 27000000.25D;
    
    System.out.println(numberFormat.format(secondAmount));
    System.out.println(numberFormat.format(fourthAmount));
    

    当将numberFormat.setGroupingUsed(false);注释掉或者修改为numberFormat.setGroupingUsed(true);时,输出结果就变为了:

    3.2 方案二:使用BigDecimal(推荐)

    使用BigDecimal的方法如下所示:

    double secondAmount = 27000000D;
    double fourthAmount = 27000000.25D;
    
    System.out.println(new BigDecimal(String.valueOf(secondAmount)).setScale(2,RoundingMode.HALF_UP).toString());
    System.out.println(new BigDecimal(String.valueOf(fourthAmount)).setScale(2,RoundingMode.HALF_UP).toString());
    

    相比而言,我更推荐使用BigDecimal的这种方案。

    关于BigDecimal的更多用法,可以查看我写的另一篇博客:Java BigDecimal使用指南


  • 相关阅读:
    弄清变量名字空间
    Perl中文编码的处理
    了解魔符的含义
    Log::Minimal 小型可定制的log模块
    Perl – 文件测试操作符
    在源代码中使用Unicode字符
    editplus乱码charset的奇怪问题
    ASP.NET程序中常用代码汇总(一)
    ASP.NET程序中常用代码汇总(三)
    ASP.NET程序中常用代码汇总(二)
  • 原文地址:https://www.cnblogs.com/zwwhnly/p/13868059.html
Copyright © 2011-2022 走看看