如何从字符串解析十进制而不损失精度?

时间:2019-04-25 07:33:17

标签: ios swift decimal nsdecimalnumber

我正在处理我的应用程序中的金额,因此精度显然非常重要,并且Decimal / NSDecimal数字应该是迅速处理金钱的方式。我想在保持精度的同时在Decimal和String之间转换。 (解析API响应,存储到数据库...)

这是我当前的解决方案:

private let dotLocaleDictionary = [NSLocale.Key.decimalSeparator: "."]

func apiStringToDecimal(_ string: String) -> Decimal? {
    let number = NSDecimalNumber(string: string, locale: dotLocaleDictionary) as Decimal
    return number.isNaN ? nil : number
}

func decimalToApiString(_ decimal: Decimal) -> String {
    return (decimal as NSDecimalNumber).description(withLocale: dotLocaleDictionary)
}

语言环境字典负责小数点分隔符,我希望字符串初始化程序精确无误。但是,在操场上玩耍时,我注意到在输入字符串一定长度后,所得到的数字会被截断。

我还尝试使用数字格式化程序解析字符串,但结果更糟。

我的游乐场代码:

let small = "1.135160000500009000100020003000400050061111"
let big = "1234567890123456789000100020003000400061111"
let medium = "123456789123456789.0001000200030004000511111"
let negative = "-123456789123456789.0001000200030004000511111"
let numericStrings = [small, big, medium, negative]

numericStrings.forEach {
    print($0)
    guard let decimal = apiStringToDecimal($0) else {
        print("Failed to parse decimal from: \($0)")
        print("--------------------")
        return
    }

    print(decimal)
    let string = decimalToApiString(decimal)
    print($0 == string ? "match" : "mismatch")

    print("--------------------")
}

let max = Decimal.greatestFiniteMagnitude
print("MAX decimal: \(max)")
print((max as NSDecimalNumber).description(withLocale: dotLocaleDictionary))
print(decimalToApiString(max))

及其产生的输出:

1.135160000500009000100020003000400050061111
1.13516000050000900010002000300040005006
mismatch
--------------------
1234567890123456789000100020003000400061111
1234567890123456789000100020003000400060000
mismatch
--------------------
123456789123456789.0001000200030004000511111
123456789123456789.000100020003000400051
mismatch
--------------------
-123456789123456789.0001000200030004000511111
-123456789123456789.000100020003000400051
mismatch
--------------------
MAX decimal: 3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000

从十进制转换为字符串似乎可以正常工作,将字符串转换为十进制似乎是个问题。 由于我的数字小于最大十进制数,因此我认为我的测试数字不会太大。 我很确定我已经排除了打印问题,并且已经在实际设备上运行了代码,所以这不是游乐场的问题。

我做错什么了吗?有没有更好的方法可以快速在小数和字符串之间转换?

还是字符串太长? (我没有找到任何文档/讨论为什么不起作用)

1 个答案:

答案 0 :(得分:3)

来自NSDecimalNumber

  

NSDecimalNumber是NSNumber的不变子类,它提供了一个面向对象的包装器,可以执行10进制算法。一个实例可以表示任何可以表示为尾数x 10 ^指数的数字,其中尾数是不超过38位的十进制整数,指数是从–128到127的整数。

(重点是我的)

尽管NSDecimalNumber提供的十进制算术精度很高,但是精度有极限。

您可能需要一个自定义库,例如BigNum