结果是0.2,用JS这样计算也会出现类似的精度问题,可为什么会出现这么匪夷所思的结果呢? 这是由于浮点数(Float)的存储规则导致的,因为Float和Double都是浮点数,都有取值范围,都有精度范围。浮点数与通常使用的小数不同,使用中往往难以确定。我们先来看十进制0.2如何转换成二进制(使用乘2取整,顺序排法),我们发现0.2不能用二进制准确表示,在二进制世界里这是个无限循环的小数,常见的问题是定义了一个浮点数, 经过一系列的计算, 它本来应该等于某个确定值, 但实际上并不是!
解决这种金额的精确计算,我们可以使用BigDecimal,BigDecimal类是专门为解决浮点数无法精确计算而设计的,并且提供了常用的算术运算方法。特别是与数据库Decimal映射时,BigDecimal是最优方案。针对mysql数据库,BigDecimal在进行入库时,数据库可选择decimal类型,长度可以自定义,如18;小数点我们项目中用的是2, 保留2位小数。此外还要注意的就是默认值,一定写成0.00,不要用默认的NULL,否则在进行加减排序等操作时,会带来转换的麻烦!
系统之间需要传递金额类型的数据时,可用字符串类型进行系统间的交互。不过在金额字符串转换成金额数值的时候要注意,否则可能会出现错误。比如下面这个例子:
我们希望得到金额数值,结果却是用科学计数法表示的字符串,原因是浮点型数据位数超过10位之后,数据变成科学计数法显示。解决这个问题可用下列方法:
在系统页面显示时我们需要对金额数据格式化输出,比如我们常见的金额数据千分位显示,可用NumberFormat进行格式化。与 DateFormat 类似,NumberFormat 是一个抽象类。您永远不会创建它的实例,相反您总是使用它的子类。虽然可以通过子类的构造函数直接创建子类,不过NumberFormat 类提供了一系列 getXXXInstance() 方法,用以获得不同类型的数值类的特定地区版本。如果做一些国际化的系统,也可以用此类做一些国际化的输出。
在系统列表页面显示金额数据时建议采用右对齐的方式,方便用户很直观的比较金额数据的大小。
DecimalFormat
DecimalFormat 是 NumberFormat 的一个具体子类,用于格式化十进制数字。该类设计有各种功能,使其能够分析和格式化任意语言环境中的数,包括对西方语言、阿拉伯语和印度语数字的支持。它还支持不同类型的数,包括整数 (123)、定点数 (123.4)、科学记数法表示的数 (1.23E4)、百分数 (12%) 和金额 ($123)。所有这些内容都可以本地化。
DecimalFormat 包含一个模式 和一组符号
符号含义:
符号 | 位置 | 本地化? | 含义 |
0 |
数字 | 是 | 阿拉伯数字 |
# |
数字字 | 是 | 阿拉伯数字,如果不存在则显示为空 |
. |
数字 | 是 | 小数分隔符或货币小数分隔符 |
- |
数字 | 是 | 减号 |
, |
数字 | 是 | 分组分隔符 |
E |
数字 | 是 | 分隔科学计数法中的尾数和指数。在前缀或后缀中无需加引号。 |
; |
子模式边界 | 是 | 分隔正数和负数子模式 |
% |
前缀或后缀 | 是 | 乘以 100 并显示为百分数 |
/u2030 |
前缀或后缀 | 是 | 乘以 1000 并显示为千分数 |
¤ (/u00A4 ) |
前缀或后缀 | 否 | 货币记号,由货币符号替换。如果两个同时出现,则用国际货币符号替换。如果出现在某个模式中,则使用货币小数分隔符,而不使用小数分隔符。 |
' |
前缀或后缀 | 否 | 用于在前缀或或后缀中为特殊字符加引号,例如 "'#'#" 将 123 格式化为 "#123" 。要创建单引号本身,请连续使用两个单引号:"# o''clock" 。 |
例子:
DecimalFormat df1 = new DecimalFormat("0.0");
DecimalFormat df2 = new DecimalFormat("#.#");
DecimalFormat df3 = new DecimalFormat("000.000");
DecimalFormat df4 = new DecimalFormat("###.###");
System.out.println(df1.format(12.34));
System.out.println(df2.format(12.34));
System.out.println(df3.format(12.34));
System.out.println(df4.format(12.34));
运行结果:
12.3
12.3
012.340
12.34
DecimalFormat format = new DecimalFormat("###,####.000");
System.out.println(format.format(111111123456.1227222));
Locale.setDefault(Locale.US);
DecimalFormat usFormat = new DecimalFormat("###,###.000");
System.out.println(usFormat.format(111111123456.1227222));
DecimalFormat addPattenFormat = new DecimalFormat();
addPattenFormat.applyPattern("##,###.000");
System.out.println(addPattenFormat.format(111111123456.1227));
DecimalFormat zhiFormat = new DecimalFormat();
zhiFormat.applyPattern("0.000E0000");
System.out.println(zhiFormat.format(10000));
System.out.println(zhiFormat.format(12345678.345));
DecimalFormat percentFormat = new DecimalFormat();
percentFormat.applyPattern("#0.000%");
System.out.println(percentFormat.format(0.3052222));
运行结果 :
1111,1112,3456.123
111,111,123,456.123
111,111,123,456.123
1.000E0004
1.235E0007
30.522%
如果使用具有多个分组字符的模式,则最后一个分隔符和整数结尾之间的间隔才是使用的分组大小。