Go中bool、整数、浮点、复数、string、uintptr及由它们构成的数组和结构体(字段全可比较)可直接用==/!=比较;slice、map、func不可比较;字符串逐字节比较,不归一化;结构体要求所有字段可比较;浮点数应避免==,改用误差范围判断。

Go 里哪些基本类型能直接用 == 和 != 比较
Go 中只有「可比较类型」才能使用 == 或 !=。基本类型中,bool、所有整数类型(int/int8/.../uint64)、浮点型(float32/float64)、复数型(complex64/complex128)、string、uintptr、以及由这些类型构成的数组和结构体(且所有字段都可比较),都支持直接比较。
常见误用:试图用 == 比较两个 slice、map 或 func 类型变量——这会编译报错:invalid operation: cannot compare ... (operator == not defined on ...)。
-
[]int{1,2} == []int{1,2}→ 编译失败 -
map[string]int{"a": 1} == map[string]int{"a": 1}→ 编译失败 -
func(){} == func(){}→ 编译失败
字符串比较要注意 Unicode 归一化吗
Go 的 string 是字节序列,== 做的是逐字节比较,不涉及 Unicode 归一化或大小写折叠。这意味着:
-
"café" == "cafe\u0301"返回false(即使视觉相同,UTF-8 编码不同) -
"ABC" == "abc"返回false(大小写敏感) - 需要语义等价时,必须显式用
golang.org/x/text/unicode/norm归一化,或用strings.EqualFold做大小写不敏感比较
import "strings"s1 := "HELLO" s2 := "hello" fmt.Println(s1 == s2) // false fmt.Println(strings.EqualFold(s1, s2)) // true
结构体比较的隐含限制
结构体能否用 == 取决于其所有字段是否可比较。哪怕只有一个不可比较字段(比如内嵌了 map 或 slice),整个结构体就不可比较。
立即学习“go语言免费学习笔记(深入)”;
-
type A struct{ X int; Y []string }→ 不能用== -
type B struct{ X int; Y string }→ 可以用== - 空结构体
struct{}是可比较的,且所有实例彼此相等(struct{}{} == struct{}{}为true)
注意:结构体比较是浅层的,不会递归检查指针指向的内容;如果字段是指针,只比地址值,不是比指针所指对象。
浮点数用 == 安全吗
不安全。由于精度丢失和舍入误差,直接用 == 比较 float32 或 float64 极易出错。
-
0.1 + 0.2 == 0.3→false(IEEE 754 表示问题) - 应改用误差范围判断:
math.Abs(a - b) -
标准库无内置浮点相等函数,需自己封装或使用第三方如
github.com/google/go-cmp/cmp(带选项支持浮点容差)
import "math"func floatEqual(a, b, epsilon float64) bool { return math.Abs(a-b) < epsilon }
x, y := 0.1+0.2, 0.3 fmt.Println(floatEqual(x, y, 1e-9)) // true
结构体或切片里含浮点字段时,也得小心——它们整体仍可比较(只要字段类型允许),但结果可能不符合数学直觉。










