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

问题描述

我想在 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 的原因。

解决方法

更新:好像这个方案还是有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 为我指引正确的方向!