最可靠的方式是用 reflect.Value.kind() == reflect.Ptr 判断指针类型;需先检查 Kind 是否为 reflect.Ptr 再调 Isnil(),且仅当 Kind 为 reflect.Ptr 且 !IsNil() 时才可安全调 Elem()。

直接看 Kind() 是否等于 reflect.Ptr,这是最可靠、最常用的判断方式。
用 reflect.Value.Kind() 判断指针类型
go 反射中,Kind() 返回的是底层类型“类别”,和是否带指针符号 * 严格对应。只要变量本身是指针(哪怕指向 nil),它的 Kind 就是 reflect.Ptr。
- 对值调用
reflect.ValueOf(v).Kind() == reflect.Ptr即可判断v是否为指针类型 - 对类型调用
reflect.typeof(v).Kind() == reflect.Ptr效果一致,适合做参数类型校验等静态检查场景 -
Name()不适用——它只对命名类型(如type Myint int)返回名字,对*int这类无名指针类型返回空字符串
package main import ( "fmt" "reflect" ) func main() { var a *int var b int var c *Struct{ X int } fmt.Println(reflect.ValueOf(a).Kind() == reflect.Ptr) // true fmt.Println(reflect.ValueOf(b).Kind() == reflect.Ptr) // false fmt.Println(reflect.ValueOf(c).Kind() == reflect.Ptr) // true }
注意 IsNil() 的调用前提
IsNil() 不能随便调,它只对某些引用类型合法,否则会 panic。必须先确认 Kind 是允许的类型,再调用。
- 允许调用
IsNil()的Kind:reflect.Ptr、reflect.map、reflect.Slice、reflect.Chan、reflect.Func、reflect.Interface - 如果
v.Kind() != reflect.Ptr却强行调v.IsNil(),运行时 panic - nil 指针的
Value仍是有效值(IsValid()返回true),但IsNil()返回true
var p *int = nil v := reflect.ValueOf(p) if v.Kind() == reflect.Ptr { fmt.Println("IsNil:", v.IsNil()) // true }
解引用前务必检查 IsNil()
调用 Elem() 获取指针所指的值时,若原指针为 nil,会 panic。这不是“意外”,而是 Go 反射的明确设计约束。
- 只有
v.Kind() == reflect.Ptr && !v.IsNil()时,才安全调用v.Elem() - 即使你确定传入的是指针,也建议加判空——尤其在泛型/框架代码中,输入不可控
-
v.Elem().Kind()才是你真正想操作的底层类型(比如reflect.Int、reflect.Struct)
if v.Kind() == reflect.Ptr && !v.IsNil() { elem := v.Elem() fmt.Println("指向类型 Kind:", elem.Kind()) // e.g. int } else { fmt.Println("无法解引用:非指针或 nil") }
最容易被忽略的一点:反射值是否可修改,和它是不是指针类型无关;而和你传给 reflect.ValueOf() 的是值还是地址有关。比如 reflect.ValueOf(&x) 才能修改 x,而 reflect.ValueOf(x) 即使 x 是指针,拿到的也只是那个指针值的拷贝,无法改原始变量。