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.Type,对指针变量调用时,得到的是该指针的类型(如 *string),而非它指向的类型。
t.Kind() == reflect.Ptr
t.Elem(),它返回被指向类型的 reflect.Type
nil 指针,reflect.TypeOf(nil) 返回 nil,需先判空示例:
var p *int = new(int)reflect.ValueOf 对指针返回一个可寻址的 reflect.Value,支持取地址、解引用、设置值(前提是原值可寻址)。
v.Kind() == reflect.Ptr 判断是否为指针值
v.IsNil() 安全检测 nil 指针(比直接解引用更安全)v.Elem() 获取指向的值(返回 reflect.Value),相当于 *p
v.CanInterface() 和 v.Interface() 可还原为原始指针(但不能对不可寻址的指针做此操作)示例:
p := &struct{ Name string }{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
示例(安全修改):
x := 42