Files
the-way-to-go_ZH_CN/eBook/09.4.md
Haigang Zhou 471d59af32 第九章修改 (#841)
Co-authored-by: Joe Chen <jc@unknwon.io>
2022-05-11 02:04:17 +08:00

55 lines
2.4 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

# 9.4 精密计算和 big 包
我们知道有些时候通过编程的方式去进行计算是不精确的。如果你使用 Go 语言中的 `float64` 类型进行浮点运算,返回结果将精确到 15 位,足以满足大多数的任务。当对超出 `int64` 或者 `uint64` 类型这样的大数进行计算时,如果对精度没有要求,`float32` 或者 `float64` 可以胜任,但如果对精度有严格要求的时候,我们不能使用浮点数,在内存中它们只能被近似的表示。
对于整数的高精度计算 Go 语言中提供了 `big` 包,被包含在 `math` 包下:有用来表示大整数的 `big.Int` 和表示大有理数的 `big.Rat` 类型(可以表示为 2/5 或 3.1416 这样的分数,而不是无理数或 π)。这些类型可以实现任意位类型的数字,只要内存足够大。缺点是更大的内存和处理开销使它们使用起来要比内置的数字类型慢很多。
大的整型数字是通过 `big.NewInt(n)` 来构造的,其中 `n``int64` 类型整数。而大有理数是通过 `big.NewRat(n, d)` 方法构造。`n`(分子)和 `d`(分母)都是 `int64` 型整数。因为 Go 语言不支持运算符重载,所以所有大数字类型都有像是 `Add()``Mul()` 这样的方法。它们作用于作为 receiver 的整数和有理数,大多数情况下它们修改 receiver 并以 receiver 作为返回结果。因为没有必要创建 `big.Int` 类型的临时变量来存放中间结果,所以运算可以被链式地调用,并节省内存。
示例 9.2 [big.go](examples/chapter_9/big.go)
``` go
// big.go
package main
import (
"fmt"
"math"
"math/big"
)
func main() {
// Here are some calculations with bigInts:
im := big.NewInt(math.MaxInt64)
in := im
io := big.NewInt(1956)
ip := big.NewInt(1)
ip.Mul(im, in).Add(ip, im).Div(ip, io)
fmt.Printf("Big Int: %v\n", ip)
// Here are some calculations with bigInts:
rm := big.NewRat(math.MaxInt64, 1956)
rn := big.NewRat(-1956, math.MaxInt64)
ro := big.NewRat(19, 56)
rp := big.NewRat(1111, 2222)
rq := big.NewRat(1, 1)
rq.Mul(rm, rn).Add(rq, ro).Mul(rq, rp)
fmt.Printf("Big Rat: %v\n", rq)
}
/* Output:
Big Int: 43492122561469640008497075573153004
Big Rat: -37/112
*/
```
输出结果:
Big Int: 43492122561469640008497075573153004
Big Rat: -37/112
## 链接
- [目录](directory.md)
- 上一节:[锁和 sync 包](09.3.md)
- 下一节:[自定义包和可见性](09.5.md)