AlamoFire responseJSON 十进制精度

AlamoFire responseJSON decimal precision

我正在使用 AlamoFire 调用我的网络服务:

ApiManager.manager.request(.GET, webServiceCallUrl,  parameters: ["id": 123])
        .validate()
        .responseJSON { response in
            switch response.result {
            case .Success:
                print(response.result.value!)

                //...

            case .Failure:
                //...
            }
    }

我的网络服务return如下JSON:

{
    //...
    "InvoiceLines": [{
        "Amount": 0.94
    }]
}

Alamofire 将其视为双精度而不是小数,因此在输出控制台中我得到:

{
    //...
    InvoiceLines =     (
                {
            Amount = "0.9399999999999999";
        }
    );
}

这会在我的代码中进一步导致舍入错误。

我在服务器上使用 Fiddler 检查 Web 服务 JSON 响应以确认它是 returning 0.94。因此,我可以排除服务器的问题,并怀疑 responseJSON 导致了我的问题。

如何将货币值设为 return 作为正确的 NSDecimalNumber 值?


吉姆 answer/comments 之后的额外信息:

var stringToTest = "{\"Amount\":0.94}"
var data = stringToTest.dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false)
var object = try NSJSONSerialization.JSONObjectWithData(data!, options: opt)

var a = object["Amount"]
print(a) //"Optional(0.9399999999999999)"

var b = NSDecimalNumber(decimal: (object["Amount"] as! NSNumber).decimalValue)
print(b) //"0.9399999999999999"

如果我在 JSON 中将值作为字符串传递,那么我可以获得我想要的结果:

var stringToTest = "{\"Amount\":\"0.94\"}"
var data = stringToTest.dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false)
var object = try NSJSONSerialization.JSONObjectWithData(data!, options: opt)

var c = NSDecimalNumber(string: (object["Amount"] as! String))
print(c) //0.94

但是我不想对 API 进行更改,因此需要一个使 JSON 保持相同格式的解决方案。我唯一的选择是每次都四舍五入吗?这似乎是一种错误的做事方式,并可能在未来引发舍入问题。

小数值本身就是浮点数,JSON 规范特别允许高指数值,您可能会认为这些值是 "real" 浮点数:

http://www.ecma-international.org/publications/files/ECMA-ST/ECMA-404.pdf 第 8 节 "numbers"

您所描述的 "decimal" 是十进制浮点数,其中您有一个整数和一个以 10 为底的指数,即在本例中为 94 * 10^-2。很少有系统支持这种存储格式,而是使用二进制浮点数,它是一个整数和一个 base2 指数。由于 2 不是 5 的因数,十分之一、百分之一等不能用二进制浮点数精确表示。这意味着内部表示总是有点偏差,但如果您尝试进行比较或打印,它会知道精度并且会按您预期的方式工作。

var bar = 0.94 // 0.939999999999999
bar == 0.94 // true
print(bar) // "0.94\n"

您在其他代码中遇到问题的地方将是由于精度问题的复杂化。在十进制浮点数中,如果你尝试做 1/3 + 1/3 - 2/3 你会得到 0.333333 + 0.333333 - 0.666667 = -0.000001 != 0,而在二进制浮点数中你会遇到同样的问题。这是因为经过一次或多次运算后,该值将超出原始数字转换的精度范围,因此按字面意思。

如果您遇到这种情况,数据实际上是固定点(例如大多数情况下的货币值),最安全的方法是将其相乘,然后只使用整数值,直到您必须显示为止,如在:

var bar = 0.94 * 100
var foo = bar * 5 // multiply price by 5
print(bar / 100)

或者,由于双精度浮点数的基础精度非常高,远远超过所需的1/100,所以在进行比较或输出时可以直接取整

var bar = 0.94
var foo = bar * 5 // multiply price by 5
print(round(bar*100) / 100)