计算货币始终是一个棘手的事情。将money表示为float64可能很诱人,但这会导致计算时出现一些令人讨厌的舍入错误。出于这个原因,最好以美分来计算货币并将其存储为Int64。
从表单,命令行或其他来源收集用户输入时,钱通常以美元形式表示。 因此,最好将其视为字符串,并将该字符串直接转换为便士而不进行浮点转换。 本节将介绍将货币的字符串表示形式转换为int64(便士)并再次返回的方法(大家在进行微信支付开发的时候可以看到,结算单位为分,传递的金额为整数)。
实践
1.建立dollars.go:
package currency
import (
    "errors"
    "strconv"
    "strings"
)
// ConvertStringDollarsToPennies 接收美元字符串并转换为int64
func ConvertStringDollarsToPennies(amount string) (int64, error) {
    // 检查传入参数是否合法
    _, err := strconv.ParseFloat(amount, 64)
    if err != nil {
        return 0, err
    }
    // 以"."进行分割
    groups := strings.Split(amount, ".")
    // 如果没有"."则取切片中的第一个元素
    result := groups[0]
    r := ""
    // 处理"."后的数据
    if len(groups) == 2 {
        if len(groups[1]) != 2 {
            return 0, errors.New("invalid cents")
        }
        r = groups[1]
        if len(r) > 2 {
            r = r[:2]
        }
    }
    // 填充0
    for len(r) < 2 {
        r += "0"
    }
    result += r
    // 转换为int
    return strconv.ParseInt(result, 10, 64)
}
2.建立pennies.go:
package currency
import (
    "strconv"
)
// ConvertPenniesToDollarString 与上面的例子类似 这是将操作方式逆转
func ConvertPenniesToDollarString(amount int64) string {
    result := strconv.FormatInt(amount, 10)
    negative := false
    if result[0] == '-' {
        result = result[1:]
        negative = true
    }
    for len(result) < 3 {
        result = "0" + result
    }
    length := len(result)
    result = result[0:length-2] + "." + result[length-2:]
    if negative {
        result = "-" + result
    }
    return result
}
3.建立main.go:
package main
import (
    "fmt"
    "github.com/agtorre/go-cookbook/chapter3/currency"
)
func main() {
    userInput := "15.93"
    pennies, err := currency.ConvertStringDollarsToPennies(userInput)
    if err != nil {
        panic(err)
    }
    fmt.Printf("User input converted to %d pennies\n", pennies)
    pennies += 15
    dollars := currency.ConvertPenniesToDollarString(pennies)
    fmt.Printf("Added 15 cents, new values is %s dollars\n", dollars)
}
4.这会输出:
User input converted to 1593 pennies
Added 15 cents, new values is 16.08 dollars说明
本节使用strconv和strings包来转换字符串格式的美元和int64中格式的便士。其中转换为float64仅作为验证之用。
strconv.ParseInt和strconv.FormatInt函数对于转换为int64和字符串非常有用。我们还利用了Go字符串可以根据需要轻松添加和切片的特性。
最后编辑: kuteng  文档更新时间: 2021-01-03 15:03   作者:kuteng