Go中手写绝对值函数有两种方式:位运算法(如(x^mask)+mask,高效但仅适用于int且需防溢出)和条件判断法(if x

在 Go 语言中,不使用 math.Abs,可以通过位运算或条件判断手写绝对值函数。核心思路是:对有符号整数,负数取反加 1 即得补码对应的正数;但直接操作符号位更高效且不依赖类型大小。
用位运算(推荐,适用于 int)
利用 Go 中有符号整数采用补码表示的特性:对负数 x,^x + 1 就是其绝对值;而 x >= 0 时直接返回 x。可合并为一行无分支表达式:
注意:该方法仅适用于 int(平台相关,通常是 int64 或 int32),且需确保无溢出(math.MinInt64 的绝对值无法表示,会溢出)。
- 提取符号位:对
int类型,右移 63 位(int64)或 31 位(int32)得到全 1(负数)或全 0(非负)掩码 - 用掩码做异或和加法:
(x ^ mask) + mask,即可实现“负数取反加 1,非负数不变”
示例(适配 int64):
立即学习“go语言免费学习笔记(深入)”;
func abs(x int64) int64 {
mask := x >> 63 // 符号位扩展为全 1(负)或全 0(正)
return (x ^ mask) + mask
}
用 if 判断(最直观,兼容所有数字类型)
不依赖底层表示,语义清晰,适用于 int、int8、float64 等任意可比较类型:
- 如果
x ,返回-x - 否则返回
x
示例:
func abs(x int) int {
if x < 0 {
return -x
}
return x
}
泛型版本(Go 1.18+)可统一处理多种类型:
func Abs[T constraints.Signed | constraints.Float](x T) T {
if x < 0 {
return -x
}
return x
}
注意事项
不要对 uint 类型求绝对值 —— 无符号数恒 ≥ 0,直接返回即可。
浮点数慎用位运算 —— 因 IEEE 754 表示复杂,建议用 if x 更安全。
最小负值问题:如 int64(-9223372036854775808) 取绝对值会溢出,结果仍是它自己(补码下无法表示对应正数)。
基本上就这些。位运算是技巧性写法,适合性能敏感且确定输入范围的场景;普通逻辑判断更易读、健壮、通用。










