1.踩坑经历
上周,一个用户反馈他创建的某个销售单无法打开,但其余销售单都可以正常打开,当时查看了生产环境的ERROR日志,发现抛了这样的异常:java.lang.NumberFormatException:Forinputstring:E。
相信大家对这个异常都不陌生,很显然,是因为将字符串转换为数字时抛出的,比如下面这样:
但仔细查看了用户报错的单据,也没有发现哪里有输入“E”这样的字符串(请原谅我第一时间没有想到是科学计数法造成的,哈哈),最后把生产环境的这条数据插入到了开发环境中,定位到原来是因为将金额转换为大写时导致的,报错的关键代码如下所示:
StringtotalAmountStr=String.valueOf(totalAmount/.0);StringamountCN=MoneyUtils.toChinese(totalAmountStr);其中totalAmount是一个long类型的变量,之所以除以.0,是因为我们数据库中存储金额都是按分为单位存储的(相信很多小伙伴也是这么存储的),第2行代码主要是为了将金额转换为大写,比如将.50转换为壹拾万零伍仟元伍角。
用户报错的那个单据,totalAmount为万,转换为分就是:000000,执行完totalAmount/.0,输出结果竟然是2.7E7,而不是预期的0000,因此导致了异常:java.lang.NumberFormatException:Forinputstring:E。
最后的解决方案是将金额转换为BigDecimal来处理,代码修改为如下所示:
StringtotalAmountStr=newBigDecimal(String.valueOf(totalAmount)).divide(newBigDecimal(),2,RoundingMode.HALF_UP).toString();StringamountCN=MoneyUtils.toChinese(totalAmountStr);2.原因分析
在Java中,当浮点数(float、double)的整数部分达到8位及以上,会以科学计数法表示,如下所示:
doublefirstAmount=000D;doublesecondAmount=0000D;doublethirdAmount=000.25D;doublefourthAmount=0000.25D;System.out.println(firstAmount);System.out.println(secondAmount);System.out.println(thirdAmount);System.out.println(fourthAmount);
默默数了下,整数部分8位的话,都是千万级别了,估计遇到这个问题的用户很豪,哈哈。
所以使用double来表示金额,当金额遇到科学计数法时,就会显示不正常、甚至造成一些意想不到的异常。
3.解决方案
如果不想用科学计数法显示,而是显示金额本身,有以下2种解决方案:
使用NumberFormat使用BigDecimal3.1方案一:使用NumberFormat
使用NumberFormat的方法如下所示:
NumberFormatnumberFormat=NumberFormat.getInstance();numberFormat.setGroupingUsed(false);doublesecondAmount=0000D;doublefourthAmount=0000.25D;System.out.println(numberFormat.format(secondAmount));System.out.println(numberFormat.format(fourthAmount));
当将numberFormat.setGroupingUsed(false);注释掉或者修改为numberFormat.setGroupingUsed(true);时,输出结果就变为了:
3.2方案二:使用BigDecimal(推荐)
使用BigDecimal的方法如下所示:
doublesecondAmount=0000D;doublefourthAmount=0000.25D;System.out.println(newBigDecimal(String.valueOf(secondAmount)).setScale(2,RoundingMode.HALF_UP).toString());System.out.println(newBigDecimal(String.valueOf(fourthAmount)).setScale(2,RoundingMode.HALF_UP).toString());
相比而言,我更推荐使用BigDecimal的这种方案。
关于BigDecimal的更多用法,可以查看我写的另一篇博客:JavaBigDecimal使用指南。