当前位置:首页 > 开发 > 开源软件 > 正文

json-lib对于浮点数出现精度问题

发表于: 2014-07-07   作者:dien   来源:转载   浏览次数:
摘要: 项目使用json-lib-2.4-jdk15.jar进行json<->pojo之间转换,今天发现一个悲剧问题: String jsonString = "[{\"amount\":670526.32},{\"amount\":29473.68}]"; JSONObject jsonObject = JSONObje
项目使用json-lib-2.4-jdk15.jar进行json<->pojo之间转换,今天发现一个悲剧问题:
String jsonString = "[{\"amount\":670526.32},{\"amount\":29473.68}]";
JSONObject jsonObject = JSONObject.fromObject(jsonString);

结果发现第一个数值转换后为670526.3,精度出现问题,网上搜索一番,没找到原因,蛮试试降低版本json-lib-2.3-jdk15.jar,竟然正确,看来这小版本变动大有文章,反编译看源码,先看看2.4的
JSONObject.java
private static JSONObject _fromJSONTokener(JSONTokener tokener, JsonConfig jsonConfig)
  {
......
v = tokener.nextValue(jsonConfig);
......
}

JSONTokener.java
public Object nextValue(JsonConfig jsonConfig)
  {
......
if (((b >= '0') && (b <= '9')) || ((((b == '.') || (b == '-') || (b == '+'))) && ((
      (b != '0') || (
      (s.length() > 2) && (((s.charAt(1) == 'x') || (s.charAt(1) == 'X'))))))));
    try
    {
      return new Integer(Integer.parseInt(s.substring(2), 16));
    }
    catch (Exception e)
    {
      try
      {
        return new Integer(Integer.parseInt(s, 8));
      }
      catch (Exception e)
      {
        try
        {
          return NumberUtils.createNumber(s);
        } catch (Exception e) {
          return s;
        }

        if ((JSONUtils.isFunctionHeader(s)) || (JSONUtils.isFunction(s)))
          return s;
      }
......
}

NumberUtils.java
  public static Number createNumber(String str)
    throws NumberFormatException
  {
......
try
    {
      return createInteger(str);
    }
    catch (NumberFormatException allZeros)
    {
      try {
        return createLong(str);
      }
      catch (NumberFormatException nfe)
      {
        return createBigInteger(str);
      }

      boolean allZeros = (isAllZeros(mant)) && (isAllZeros(exp));
      try {
        Float f = createFloat(str);
        if ((!(f.isInfinite())) && (((f.floatValue() != 0.0F) || (allZeros))))
          return f;
      }
      catch (NumberFormatException nfe)
      {
      }
      try {
        Double d = createDouble(str);
        if ((!(d.isInfinite())) && (((d.doubleValue() != 0.0D) || (allZeros))))
          return d;
      }
      catch (NumberFormatException d)
      {
      }
    }
    return createBigDecimal(str);
}

以上看出,对于浮点数字,会先使用createFloat()进行转换,此处也是导致精度出现问题的关键
2.3代码:
JSONTokener.java
public Object nextValue(JsonConfig jsonConfig)
  {
......
if (((b >= '0') && (b <= '9')) || ((((b == '.') || (b == '-') || (b == '+'))) && ((
      (b != '0') || (
      (s.length() > 2) && (((s.charAt(1) == 'x') || (s.charAt(1) == 'X'))))))));
    try
    {
      return new Integer(Integer.parseInt(s.substring(2), 16));
    }
    catch (Exception e)
    {
      try
      {
        return new Integer(Integer.parseInt(s, 8));
      }
      catch (Exception e)
      {
        try
        {
          return new Integer(s);
        } catch (Exception e) {
          try {
            return new Long(s);
          } catch (Exception f) {
            try {
              return new Double(s);
            } catch (Exception g) {
              return s;
            }

            if ((JSONUtils.isFunctionHeader(s)) || (JSONUtils.isFunction(s)))
              return s;
          }
}

以上看出,对于浮点数字,会先使用new Double(s)

最后写个main函数测试:
String a = "670526.01";
System.out.println("aa:"+Float.valueOf(a));
System.out.println("aa:"+new Double(a));

打印结果:
aa:670526.3
aa:670526.32

小版本的差异,出现了如此差异,实在不知说什么好。
目前只能降低版本,不过要多做测试;还有一种方法,全部使用字符串,如String jsonString = "[{\"amount\":\"670526.32\"},{\"amount\":\"29473.68\"}]";

json-lib对于浮点数出现精度问题

  • 0

    开心

    开心

  • 0

    板砖

    板砖

  • 0

    感动

    感动

  • 0

    有用

    有用

  • 0

    疑问

    疑问

  • 0

    难过

    难过

  • 0

    无聊

    无聊

  • 0

    震惊

    震惊

编辑推荐
C#中的浮点数,分单精度(float)和双精度(double): float 是 System.Single 的别名,介于 -3.40
Json-lib中对于float会默认按double类型处理,因此从数据库中取出的值1.1会变成1.100000023841858 问
本篇讨论的现象可以从下面这段脚本体现出来: >>> x = 0.0 >>> for i in range(
在x86/x64体系里,由于x87 FPU硬件使用扩展双精度格式,因此必然会遇到single/double precision格式
一般来说判断时间的话,用个DateTime类型就已经够用了。但是有些情况,比如下面这种 DECLARE @DT1 D
在JS处理算法时,有时会存在精度问题。 如694.84+(-300)有精度问题,但694.64+(-300)没有精度问题。
在JS处理算法时,有时会存在精度问题。 如694.84+(-300)有精度问题,但694.64+(-300)没有精度问题。
问题来源: 前几天用户向我反映了一个问题,有一个金额字段当输入到达百万级时,个位数的精度会丢失
今天继续学习《C语言的科学和艺术》一书,今晚终于结束了第六章算法部分的课后习题,同时我也非常高
今天继续学习《C语言的科学和艺术》一书,今晚终于结束了第六章算法部分的课后习题,同时我也非常高
版权所有 IT知识库 CopyRight © 2009-2015 IT知识库 IT610.com , All Rights Reserved. 京ICP备09083238号