Go 中如何正确导入和使用 math/rand 随机数包

go 语言中,`math/rand` 是一个独立的标准库子包,并非 `math` 包的内置功能;必须显式导入 `"math/rand"` 才能使用其类型与函数,仅导入 `"math"` 无法访问 `rand`。

在 Go 的模块化设计中,包路径即包标识符,不存在“父包自动导出子包符号”的概念。虽然 math/rand 在文件系统路径上位于 math/ 目录下,但它是一个逻辑上完全独立的包,拥有自己的命名空间、导出规则和依赖关系。"math" 包本身只提供数学常量(如 math.Pi)和基础函数(如 math.Sin、math.Sqrt),不包含任何随机数相关功能

因此,以下代码会编译失败:

package main

import (
    "fmt"
    "math" // ❌ 错误:math 包中没有 rand 标识符
)

func main() {
    r := rand.New(rand.NewSource(99)) // 编译错误:undefined: rand
    fmt.Println(r)
}

✅ 正确做法是显式导入 "math/rand",并确保在代码中通过 rand. 前缀调用其导出成员:

package main

import (
    "fmt"
    "math/rand" // ✅ 正确:导入独立的随机数包
    "time"
)

func main() {
    // 使用确定性种子便于演示(生产环境建议用 time.Now().UnixNano())
    src := rand.NewSource(99)
    r := rand.New(src)

    fmt.Println(r.Intn(100)) // 输出示例:42
    fmt.Println(r.Float64()) // 输出示例:0.123456789...
}

⚠️ 注意事项:

  • Go 不支持“通配符导入”或“包聚合导入”(如 import "math/*"),每个需用包必须单独声明;
  • 自 Go 1.20 起,推荐优先使用 crypto/rand(加密安全随机数)替代 math/rand 处理敏感场景(如生成密钥、token);
  • math/rand 的默认全局实例(rand.Intn 等顶层函数)已弃用(自 Go 1.20 起标记为 Deprecated: Use the Rand type instead),应始终显式创建 *rand.Rand 实例以保证可预测性和并发安全。

总结:Go 的包机制强调显式性与最小依赖——你用什么,就导入什么。"math/rand" 不是 "math" 的子模块,而是一个同属标准库、路径命名体现分类关系的独立包。理解这一点,是写出可维护、可移植 Go 代码的基础。