Golang ParseFloat在示例中不准确

时间:2018-08-08 23:55:26

标签: parsing go

我一直在进行一个项目,在该项目中,我必须将字符串转换为uint,以确保某些money值匹配:

total, err := strconv.ParseFloat(paymentResp.Transactions[0].Amount.Total, 64)
if err != nil {
    return ctx.JSON(http.StatusBadRequest, err.Error())
}

if o.TotalPrice != uint(total*100) {
    return ctx.JSON(http.StatusBadRequest, "Unable to verify amount paid")
}

但是当我尝试对两个数字进行strconv.ParseFloat(),然后尝试将它们乘以100(以获得美分值)时,我似乎发现了一个问题。

我在这里创建了一个示例:Go Playground

f, _ := strconv.ParseFloat("79.35", 64)
fmt.Println(uint(f*100))  //7934

f2, _ := strconv.ParseFloat("149.20", 64)
fmt.Println(uint(f2*100)) //14919

在这种情况下我应该使用ParseFloat()吗?如果没有,我很想听听对此的简短解释,因为我仍然是一个学习中的程序员。

3 个答案:

答案 0 :(得分:1)

Go使用IEEE-754个二进制浮点数。浮点数不精确。不要将它们用于金融交易。使用整数。

例如,

import org.apache.spark.sql.functions._
import scala.util.Try
import scala.util.Success

val checkIntUDF = udf { (x: String) => val y = Try(x.toInt); y match { case Success(x) => true; case _ => false; } }

游乐场:https://play.golang.org/p/mGuO51QWyIv

输出:

package main

import (
    "fmt"
    "strconv"
    "strings"
)

func parseCents(s string) (int64, error) {
    n := strings.SplitN(s, ".", 3)
    if len(n) != 2 || len(n[1]) != 2 {
        err := fmt.Errorf("format error: %s", s)
        return 0, err
    }
    d, err := strconv.ParseInt(n[0], 10, 56)
    if err != nil {
        return 0, err
    }
    c, err := strconv.ParseUint(n[1], 10, 8)
    if err != nil {
        return 0, err
    }
    if d < 0 {
        c = -c
    }
    return d*100 + int64(c), nil
}

func main() {
    s := "79.35"
    fmt.Println(parseCents(s))
    s = "149.20"
    fmt.Println(parseCents(s))
    s = "-149.20"
    fmt.Println(parseCents(s))
    s = "149.2"
    fmt.Println(parseCents(s))
}

答案 1 :(得分:0)

基于@peterSO的答案,并进行了一些错误修复和增强:

https://play.golang.org/p/YcRLeEJ7lTA

package main

import (
    "fmt"
    "strconv"
    "strings"
)

func parseCents(s string) (int64, error) {
    var ds string
    var cs string

    n := strings.SplitN(s, ".", 3)
    switch len(n) {
    case 1:
        ds = n[0]
        cs = "0"
    case 2:
        ds = n[0]
        switch len(n[1]) {
        case 1:
            cs = n[1] + "0"
        case 2:
            cs = n[1]
        default:
            return 0, fmt.Errorf("invalid format:%s", s)
        }
    default:
        return 0, fmt.Errorf("invalid format:%s", s)
    }

    d, err := strconv.ParseInt(ds, 10, 0)
    if err != nil {
        return 0, err
    }

    c, err := strconv.ParseUint(cs, 10, 0)
    if err != nil {
        return 0, err
    }

    cents := d * 100

    if strings.HasPrefix(s, "-") {
        cents -= int64(c)
    } else {
        cents += int64(c)
    }

    return cents, nil
}

func main() {
    examples := map[string]int64{
        "79.35": 7935,
        "149.20": 14920,
        "-149.20": -14920,
        "149.2": 14920,
        "-0.12": -12,
        "12": 1200,
        "1.234": 0,
        "1.2.34": 0,
    }

    for s, v := range examples {
        cents, err := parseCents(s)
        fmt.Println(cents, cents == v, err)
    }
}

答案 2 :(得分:-1)

您可以拆分字符串,并将其解析为一个整数。

package main

import (
    "fmt"
    "strconv"
    "strings"
    )

func main() {
    a:=strings.Split("75.35",".")
    if len(a)>2{
    panic("badly formatted price")    
     }
    v,_:=strconv.ParseInt(a[1],10,64)
    w,_:=strconv.ParseInt(a[0],10,64)
   fmt.Println(uint(w*100+v))
}

它在此链接https://play.golang.org/p/5s_FTAKSo9M

中工作