Go 1.21+推荐使用cmp.Max和cmp.Min(支持任意可比较类型且类型必须相同),math.Max/Min仅限float64;切片需手动遍历或泛型函数实现;旧版本可用自定义函数。

Go语言标准库中没有直接提供适用于所有数字类型的math.Max和math.Min函数,这是因为Go在1.21版本之前不支持泛型重载,而math包中的Max和Min仅支持float64类型。从Go 1.21开始,标准库新增了max和min泛型函数(位于golang.org/x/exp/constraints已废弃,实际已移入cmp包),但更常用、更推荐的方式是使用Go 1.21+内置的cmp.Max和cmp.Min,或自己写简洁的比较逻辑。
使用 cmp.Max 和 cmp.Min(推荐,Go 1.21+)
Go 1.21起,标准库新增cmp包(cmp是compare缩写),其中cmp.Max和cmp.Min支持任意可比较类型(如int、int64、float64、string等),只要两个值类型相同且可比较即可。
- 需导入:
import "cmp" - 用法示例:
a, b := 15, 8 maxVal := cmp.Max(a, b) // int 类型,返回 15 minVal := cmp.Min(a, b) // 返回 8 x, y := 3.14, 2.71 maxF := cmp.Max(x, y) // float64,返回 3.14
- 注意:
cmp.Max/Min不支持混合类型(如int和int64不能混用),编译会报错
使用 math.Max 和 math.Min(仅限 float64)
math.Max和math.Min是传统方式,但只接受float64参数,传入整数需显式转换,且对NaN有特殊行为(如math.Max(1, NaN)返回NaN)。
- 需导入:
import "math" - 整数需转
float64:i, j := 10, -5 maxF := math.Max(float64(i), float64(j)) // 得 10.0
- 不适用于
int切片求最大值——必须手动遍历或用cmp辅助
对切片求最大最小值(常见需求)
标准库无MaxOfSlice之类函数,需自行遍历。配合cmp可写出类型安全、简洁的通用函数:
立即学习“go语言免费学习笔记(深入)”;
- 整数切片示例:
func maxIntSlice(s []int) (int, bool) { if len(s) == 0 { return 0, false } m := s[0] for _, v := range s[1:] { m = cmp.Max(m, v) } return m, true } - 泛型版本(Go 1.18+):
func MaxSlice[T constraints.Ordered](s []T) (T, bool) { if len(s) == 0 { var zero T; return zero, false } m := s[0] for _, v := range s[1:] { m = cmp.Max(m, v) } return m, true }(需import "constraints",但注意constraints已在Go 1.21标记为deprecated;实际可改用comparable+ 显式比较,或直接依赖cmp)
自定义简单比较函数(兼容老版本或轻量场景)
若项目仍用Go cmp,可用内联三元风格(Go无?:,但可用if表达式封装):
- 例如:
max := func(a, b int) int { if a > b { return a }; return b } min := func(a, b int) int { if a < b { return a }; return b } x := max(42, 19) // 42 - 优点:零依赖、清晰、无类型限制(每个类型写一个即可)
- 缺点:重复代码多,不适合大量类型
基本上就这些。优先用cmp.Max/Min(Go 1.21+),兼顾类型安全与简洁;老版本可用自定义函数或math配合类型转换;切片操作记得判空。不复杂但容易忽略版本和类型匹配细节。










