Go语言数学计算的秘密武器:big包详解

简介: Go语言数学计算的秘密武器:big包详解

1. big 包介绍与背景

需求背景

通常情况下,使用内置的 int 和 float64 等类型就可以满足大部分计算需求。

但是这些类型都有上限,比如 int 在 64 位系统上最大值为 9223372036854775807。

当需要处理比这更大或者更小的数时,内置类型就不够用了。

此外,浮点数计算存在精度问题,可能在某些精确场景下产生误差。

big 包的主要功能

big 包提供了以下高精度计算功能:

BigInt:无限大小的整数

BigFloat:高精度的浮点数

BigRat:高精度的有理数

可以用它们表示极其大或者极其小的整数、分数,并进行运算操作。

big 包的基本特点

值类型:BigInt 等类型使用值传递

不可变性:大数值一旦创建就不可变更

按值比较:大数值只按值比较,不相等则永不相等

准确实现:计算结果准确,不会有误差


 

2. 基本数据类型与初始化

Int 类型

big包中最基本的数据类型是Int,它可以表示任意精度的有符号整数。下面是一个Int类型的初始化示例:


import (    "fmt"    "math/big")
func main() {    num := big.NewInt(1234567890)    fmt.Println("Big Int:", num)}

Rat 类型

Rat类型表示有理数,即分数。它用两个Int类型的数值表示分子和分母。以下是一个Rat类型的初始化示例:


import (    "fmt"    "math/big")
func main() {    num1 := big.NewRat(1, 2) // 1/2    num2 := big.NewRat(3, 4) // 3/4        sum := new(big.Rat)    sum.Add(num1, num2) // 1/2 + 3/4 = 5/4        fmt.Println("Sum:", sum)}


 

3. 高精度整数操作

加法

使用Add()函数进行两个大整数的加法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Int)    num1.SetString("12345678901234567890", 10)        num2 := new(big.Int)    num2.SetString("98765432109876543210", 10)        sum := new(big.Int)    sum.Add(num1, num2)        fmt.Println("Sum:", sum)}

减法

使用Sub()函数进行两个大整数的减法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Int)    num1.SetString("98765432109876543210", 10)        num2 := new(big.Int)    num2.SetString("12345678901234567890", 10)        diff := new(big.Int)    diff.Sub(num1, num2)        fmt.Println("Difference:", diff)}

乘法

使用Mul()函数进行两个大整数的乘法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Int)    num1.SetString("12345678901234567890", 10)        num2 := new(big.Int)    num2.SetString("98765432109876543210", 10)        product := new(big.Int)    product.Mul(num1, num2)        fmt.Println("Product:", product)}

除法

使用Div()函数进行两个大整数的除法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Int)    num1.SetString("98765432109876543210", 10)        num2 := new(big.Int)    num2.SetString("12345678901234567890", 10)        quotient := new(big.Int)    quotient.Div(num1, num2)        fmt.Println("Quotient:", quotient)}


 

4. 高精度浮点数操作

加法

使用Add()函数进行两个大浮点数的加法操作:

减法

使用Sub()函数进行两个大浮点数的减法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Rat)    num1.SetString("3/5")        num2 := new(big.Rat)    num2.SetString("1/5")        diff := new(big.Rat)    diff.Sub(num1, num2)        fmt.Println("Difference:", diff)}

乘法

使用Mul()函数进行两个大浮点数的乘法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Rat)    num1.SetString("2/3")        num2 := new(big.Rat)    num2.SetString("3/4")        product := new(big.Rat)    product.Mul(num1, num2)        fmt.Println("Product:", product)}

除法

使用Quo()函数进行两个大浮点数的除法操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Rat)    num1.SetString("2/3")       
 num2 := new(big.Rat)    num2.SetString("1/4")        quotient := new(big.Rat)    quotient.Quo(num1, num2)        fmt.Println("Quotient:", quotient)}


 

5. 高精度数学函数

指数函数

使用Exp()函数计算大浮点数的指数函数:


import (    "fmt"    "math/big")
func main() {    base := new(big.Rat)    base.SetString("2")        exponent := new(big.Int)    exponent.SetString("10")        result := new(big.Rat)    result.Exp(base, exponent)        fmt.Println("Result:", result)}

对数函数

使用Sqrt()函数计算大浮点数的平方根:


import (    "fmt"    "math/big")
func main() {    num := new(big.Rat)    num.SetString("25")        sqrt := new(big.Rat)    sqrt.Sqrt(num)        fmt.Println("Square Root:", sqrt)}

三角函数

使用Sin(), Cos(), Tan()等函数计算大浮点数的三角函数值:



import (    "fmt"    "math/big"    "math")
func main() {    angle := new(big.Rat)    angle.SetString("45")        sinValue := new(big.Rat)    sinValue.Sin(angle)        cosValue := new(big.Rat)    cosValue.Cos(angle)        tanValue := new(big.Rat)    tanValue.Tan(angle)        fmt.Println("Sin(45°):", sinValue)    fmt.Println("Cos(45°):", cosValue)    fmt.Println("Tan(45°):", tanValue)}


 

6. 大整数的比较与格式化

比较操作

使用Cmp()函数进行两个大整数的比较操作:


import (    "fmt"    "math/big")
func main() {    num1 := new(big.Int)    num1.SetString("12345678901234567890", 10)        num2 := new(big.Int)    num2.SetString("98765432109876543210", 10)        cmpResult := num1.Cmp(num2)    switch cmpResult {    case -1:        fmt.Println("num1 < num2")    case 0:        fmt.Println("num1 = num2")    case 1:        fmt.Println("num1 > num2")    }}

格式化操作

使用String()函数将大整数格式化为字符串:


import (    "fmt"    "math/big")
func main() {    num := new(big.Int)    num.SetString("12345678901234567890", 10)        str := num.String()    fmt.Println("Formatted String:", str)}


 

7. 高级应用:大整数的 RSA 加解密

密钥生成

使用big包生成 RSA 密钥对:


import (    "crypto/rand"    "crypto/rsa"    "fmt"    "math/big")
func main() {    privateKey, err := rsa.GenerateKey(rand.Reader, 2048)    if err != nil {        fmt.Println("Error generating private key:", err)        return    }
    publicKey := &privateKey.PublicKey
    fmt.Println("Private Key:", privateKey)    fmt.Println("Public Key:", publicKey)}

加密解密

使用 RSA 公钥加密和私钥解密:


import (    "crypto/rand"    "crypto/rsa"    "fmt")
func main() {    message := []byte("Hello, RSA!")
    // 加密    publicKey, err := rsa.GenerateKey(rand.Reader, 2048)    if err != nil {        fmt.Println("Error generating public key:", err)        return    }    ciphertext, err := rsa.EncryptPKCS1v15(rand.Reader, &publicKey, message)    if err != nil {        fmt.Println("Error encrypting message:", err)        return    }
    // 解密    plaintext, err := rsa.DecryptPKCS1v15(rand.Reader, publicKey, ciphertext)    if err != nil {        fmt.Println("Error decrypting message:", err)        return    }
    fmt.Println("Original message:", string(plaintext))}


 

8. 性能优化与注意事项

在使用big包进行高精度计算时,应该注意性能和内存的消耗。

在处理大整数时,选择适当的数据类型以及避免不必要的计算可以提高程序的效率。


 

9. 总结

Go 语言大数包 big 提供了 BigInt、BigFloat 和 BigRat 三种高精度计算类型。

相比内置类型,它们可以表示非常大或者非常小的数,并进行各种精确运算。

使用 big 包可以避免整数溢出和浮点数误差。

big 包的类型与方法设计简洁高效,可以很容易地进行大数计算。

它在密码学、统计分析等需要高精度计算的领域非常实用。


目录
相关文章
|
22天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
36 7
|
22天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
22天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
97 71
|
21天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
102 67
|
2天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
29 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
16天前
|
Linux Go iOS开发
怎么禁用 vscode 中点击 go 包名时自动打开浏览器跳转到 pkg.go.dev
本文介绍了如何在 VSCode 中禁用点击 Go 包名时自动打开浏览器跳转到 pkg.go.dev 的功能。通过将 gopls 的 `ui.navigation.importShortcut` 设置为 &quot;Definition&quot;,可以实现仅跳转到定义处而不打开链接。具体操作步骤包括:打开设置、搜索 gopls、编辑 settings.json 文件并保存更改,最后重启 VSCode 使设置生效。
43 7
怎么禁用 vscode 中点击 go 包名时自动打开浏览器跳转到 pkg.go.dev
|
22天前
|
存储 Go
go语言中映射
go语言中映射
34 11
|
23天前
|
Go 索引
go语言使用range关键字
go语言使用range关键字
29 7
|
23天前
|
Go 索引
go语言修改元素
go语言修改元素
29 6
|
14天前
|
Go 数据安全/隐私保护 UED
优化Go语言中的网络连接:设置代理超时参数
优化Go语言中的网络连接:设置代理超时参数