Go中指针相等性通过==和!=比较内存地址,类型需可比较,指向同一变量或均为nil时相等,即使值相同但地址不同则不等,如p1=&a、p2=&a为true,p1=&a、p3=&b为false;不同类型指针需类型兼容或使用unsafe.Pointer转换后比较,但应避免滥用unsafe;所有指针可与nil比较,未初始化默认为nil;引用类型如切片、map的指针也仅当地址相同时才相等,即便底层数组共享。

在Golang中,判断两个指针是否相等是通过比较它们指向的内存地址来实现的。如果两个指针变量保存的是同一个地址,那么它们就相等;否则不相等。
指针相等的基本规则
Go使用==和!=操作符来判断两个指针是否相等。其核心逻辑如下:
- 两个指针对应的类型必须可比较(即属于同一类型或可以赋值兼容)
- 只有当两个指针指向相同的变量地址,或者都为nil时,才判定为相等
- 即使两个变量内容相同但地址不同,它们的指针也不相等
示例代码:
a := 42
b := 42
p1 := &a
p2 := &a // 指向同一个变量
p3 := &b // 指向另一个同值变量
p4 := (*int)(nil)
fmt.Println(p1 == p2) // true:指向同一个地址
fmt.Println(p1 == p3) // false:虽然*a == *b,但地址不同
fmt.Println(p4 == nil) // true:nil指针比较
不同类型指针之间的比较
Go允许在某些条件下跨类型比较指针,前提是这两个指针类型之间可以相互赋值。例如,具有相同底层结构的命名类型。
立即学习“go语言免费学习笔记(深入)”;
type IntAlias int
var x int = 10
var y IntAlias = 10
p := &x
q := (*IntAlias)(&y) // 强制转换类型
// fmt.Println(p == q) // 编译错误:不能直接比较*int 和 *IntAlias
// 但可以这样:
r := (*IntAlias)(unsafe.Pointer(p))
fmt.Println(r == q) // 可以比较,但需谨慎使用 unsafe
注意:涉及unsafe.Pointer的操作绕过了类型安全,应尽量避免,除非在特殊场景如系统编程或性能优化中。
nil指针的比较
所有类型的指针都可以与nil进行比较。未初始化的指针默认值为nil,比较结果为真。
var p *int
var q *[5]int
fmt.Println(p == nil) // true
fmt.Println(q == nil) // true
// fmt.Println(p == q) // 编译错误:类型不同无法比较
切片、map、func等引用类型的指针比较
像切片、map、函数等本身是引用类型,它们的指针也可以比较:
s1 := []int{1,2,3}
s2 := s1
ps1 := &s1
ps2 := &s2
fmt.Println(ps1 == ps2) // false:虽然s1和s2共享底层数组,但ps1和ps2是不同变量的地址
这说明即使两个引用类型共享数据,只要它们本身的地址不同,其指针就不相等。
基本上就这些。Go的指针相等性判断简单直接:看地址是否相同,类型是否可比较,以及是否为nil。不复杂但容易忽略细节。










