如何在 Go 中将货币值解析为 *big.Int?

时间:2021-04-01 15:21:14

标签: go parsing currency biginteger

我想在 Go 中将像“12.49”这样的字符串解析为 *big.Int。结果 *big.Int 应表示给定值中的美分数量,在本例中为 1249。以下是输入及其预期输出的更多示例:

  • “3”:300
  • “3.1”:310
  • ".19": 19

我已经尝试使用 *big.Floatits Int function,但意识到 *big.Float 不能提供任意精度。

现在我正在使用这个算法,但它看起来很脆弱 (Go Playground link):

func eurToCents(in string) *big.Int {
    missingZerosUntilCents := 2
    i := strings.Index(in, ".")
    if i > -1 {
        missingZerosUntilCents -= len(in) - i - 1
        if missingZerosUntilCents < 0 {
            panic("too many decimal places")
        }
    }
    in = strings.Replace(in, ".", "", 1)
    in += strings.Repeat("0", missingZerosUntilCents)
    out, ok := big.NewInt(0).SetString(in, 10)
    if !ok {
        panic(fmt.Sprintf("could not parse '%s' as an interger", in))
    }
    return out
}

Go 中是否有标准库函数或其他常用方法来解析货币?外部库不是一种选择。

PS:我正在解析 Nano cryptocurrency 值,它有 30 个小数位,最大值为 133,248,297.0。这就是我要求 *big.Int 而不是 uint64 的原因。

1 个答案:

答案 0 :(得分:0)

更新:好像这个方案还是有bug,因为乘法后报的结果不准确:https://play.golang.org/p/RS-DC6SeRwz


在使用 *big.Float 重新审视解决方案后,我意识到它确实工作得很好。我想我之前忘记在 SetPrec 上使用 rawPerNano。我将为 Nano 加密货币提供一个示例,因为它需要很多小数位。

这按预期工作 (Go Playground link):

func nanoToRaw(in string) *big.Int {
    f, _ := big.NewFloat(0).SetPrec(128).SetString(in)
    rawPerNano, _ := big.NewFloat(0).SetPrec(128).SetString("1000000000000000000000000000000")
    f.Mul(f, rawPerNano)
    i, _ := f.Int(big.NewInt(0))
    return i
}

感谢@hymns-for-disco 为我指引正确的方向!