Go中指针无运行时独立类型信息,但可通过reflect.TypeOf获取其类型(如*int),用t.Kind()==reflect.Ptr判断,t.Elem()获取指向类型;reflect.ValueOf支持解引用(v.Elem())、nil检测(v.IsNil())及安全修改(需v.Elem().CanSet())。

在 Go 中,指针本身没有独立的“类型信息”可直接获取(比如 *int 不是运行时的一等类型),但你可以通过 reflect 包获取其底层指向类型的元信息,或通过类型断言、接口判断等方式识别和操作指针变量。关键在于区分「编译期类型」和「运行时反射类型」。
用 reflect.TypeOf 获取指针的反射类型
reflect.TypeOf 能返回任意值的 reflect.Type,对指针变量调用时,得到的是该指针的类型(如 *string),而非它指向的类型。
- 要判断一个接口值是否为指针:检查
t.Kind() == reflect.Ptr - 要获取指针指向的类型:调用
t.Elem(),它返回被指向类型的reflect.Type - 注意:若传入的是
nil指针,reflect.TypeOf(nil)返回nil,需先判空
示例:
var p *int = new(int)t := reflect.TypeOf(p)
fmt.Println(t.Kind()) // ptr
fmt.Println(t.Elem()) // int(类型名)
fmt.Println(t.String()) // *int
用 reflect.Value 处理指针值与解引用
reflect.ValueOf 对指针返回一个可寻址的 reflect.Value,支持取地址、解引用、设置值(前提是原值可寻址)。
立即学习“go语言免费学习笔记(深入)”;
- 用
v.Kind() == reflect.Ptr判断是否为指针值 - 用
v.IsNil()安全检测nil指针(比直接解引用更安全) - 用
v.Elem()获取指向的值(返回reflect.Value),相当于*p - 用
v.CanInterface()和v.Interface()可还原为原始指针(但不能对不可寻址的指针做此操作)
示例:
p := &struct{ Name string }{Name: "Alice"}v := reflect.ValueOf(p)
if v.Kind() == reflect.Ptr && !v.IsNil() {
target := v.Elem().Interface() // 得到 *struct{ Name string } 指向的 struct 值
fmt.Printf("%+v\n", target) // {Name:"Alice"}
}
类型断言与接口判断识别指针
当变量是 interface{} 类型时,无法直接用 == 比较类型,但可通过类型断言或 reflect 判断是否为某具体指针类型。
- 精确匹配:用类型断言
v, ok := iface.(**int)判断是否为**int - 泛化判断:用
reflect.TypeOf(iface).Kind() == reflect.Ptr判断是否为任意指针 - 结合
reflect.Type.Elem()进一步判断指向的具体类型,例如:t.Elem().Name() == "MyStruct"
注意:类型断言失败会 panic(不带 , ok 形式),生产代码推荐用带布尔返回值的安全形式。
修改指针所指向的值(需满足可寻址性)
只有原始变量本身可寻址(如局部变量、结构体字段、切片元素),其指针才能被安全修改;从函数参数或 map 中取出的指针若来源不可寻址,则 v.Elem().CanSet() 返回 false。
- 先确认
v.Kind() == reflect.Ptr && !v.IsNil() && v.Elem().CanSet() - 再调用
v.Elem().Set(x),其中x是同类型且可赋值的reflect.Value - 常见错误:对常量、字面量取地址后试图修改(Go 编译器禁止,反射也会失败)
示例(安全修改):
x := 42p := &x
v := reflect.ValueOf(p)
if v.Kind() == reflect.Ptr && !v.IsNil() && v.Elem().CanSet() {
v.Elem().SetInt(100)
}
fmt.Println(x) // 输出 100










