最可靠的方式是用 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 是指针,拿到的也只是那个指针值的拷贝,无法改原始变量。










