go 解决货币计算的难题:避免浮点数陷阱

在开发的初始阶段,我们经常会遇到"浮点数精度"和"货币值表示"的问题。

那么,如何处理货币,如何存储和传递它们。

为什么是问题?

Go语言中的标准浮点类型具有一定的精度(像其他任何语言一样),你不能在货币操作中使用它们。这里有一个最简单的例子:

go 复制代码
var v1, v2 = 0.1, 0.2
fmt.Println(v1 + v2)
// 输出:0.30000000000000004

你可以计算你需要将一个值与另一个值相加多少次,才能在你的账户上获得额外的钱!但反过来也是一样 --- 在这种情况下,你只是失去了你的钱。

这不仅在对你的钱进行数学运算时有问题,而且在不同系统或服务之间传递数据时也是有问题的。

下一个问题 --- 传递你的钱

每次将你的钱从/到浮点数进行编组时,都会遇到与上述相同的问题,以及与编组器实现有关的其他问题 - json,xml,text等等...

另一个问题是四舍五入。如果你处理的是货币,你总会面临四舍五入的问题。你应该如何四舍五入你的货币值?例如 0.345 元,一般我们还是会四舍五入到 0.35 元?

我们的选择是什么?

有一些特殊的类型可用于货币的表示和计算。

Go标准库有 big.Float 类型(来自 math/big 包,表示任意精度的浮点数)。与 float32 和 float64 不同,它们具有固定的大小和精度,big.Float 允许你为数字和计算设置任意精度。

另一个不错的选择是 decimal 库 github.com/shopspring/...

关于四舍五入:

  • 1.234 => 1.23
  • 1.235 => 1.24
  • 1.236 => 1.24

例如,shopspring/decimal 提供了适当舍入值的方法。

考虑的另一个好选择是使用货币单位。这样,你就从浮点数问题转移到整数,并将一切都作为整数计算。在这里唯一使用四舍五入的地方:传递结果值。

现在让我们讨论一下在传递货币时的选择。

  • 使用货币单位 --- 我们将所有内容都传递为整数,这里没有浮点问题。只需控制值的限制,就可以了。
  • 将浮点数作为字符串传递。通常也是一个不错的选择 --- 当你将浮点数作为字符串传递时,带有所需精度(特定小数位数)的字符串,当对方读取此字符串值并将其转换回浮点数时,你就是安全的。

简单的例子

你可以在 Go Playground 上尝试一下。

go 复制代码
package main

import (
    "fmt"
    "github.com/shopspring/decimal"
)

func main() {
    a := 0.1
    b := 0.2
    c := decimal.NewFromFloat(a)
    d := decimal.NewFromFloat(b)
    fmt.Println(a, b, c.String(), d.String()) 
    fmt.Println(a + b) 
    fmt.Println(c.Add(d).String()) 
}

输出为:

go 复制代码
0.1 0.2 0.1 0.2
0.30000000000000004
0.3

结论

处理货币时 --- 使用 math/big 或一些与货币相关的库,比如 shopspring/decimal,或者只是使用货币单位,在这里不要使用浮点数。将货币作为字符串传递,或者在货币单位中传递,不要在这里使用浮点数。

相关推荐
用余生去守护7 分钟前
python报错系列(16)--pyinstaller ????????
开发语言·python
数据小爬虫@11 分钟前
利用Python爬虫快速获取商品历史价格信息
开发语言·爬虫·python
向宇it13 分钟前
【从零开始入门unity游戏开发之——C#篇25】C#面向对象动态多态——virtual、override 和 base 关键字、抽象类和抽象方法
java·开发语言·unity·c#·游戏引擎
莫名其妙小饼干30 分钟前
网上球鞋竞拍系统|Java|SSM|VUE| 前后端分离
java·开发语言·maven·mssql
十年一梦实验室39 分钟前
【C++】sophus : sim_details.hpp 实现了矩阵函数 W、其导数,以及其逆 (十七)
开发语言·c++·线性代数·矩阵
isolusion42 分钟前
Springboot的创建方式
java·spring boot·后端
最爱番茄味1 小时前
Python实例之函数基础打卡篇
开发语言·python
zjw_rp1 小时前
Spring-AOP
java·后端·spring·spring-aop
Oneforlove_twoforjob1 小时前
【Java基础面试题033】Java泛型的作用是什么?
java·开发语言
TodoCoder2 小时前
【编程思想】CopyOnWrite是如何解决高并发场景中的读写瓶颈?
java·后端·面试