直接对任意 interface{} 调用 reflect.ValueOf(i).IsNil() 会 panic,因 IsNil() 仅支持指针、切片、map、channel、func、interface 六种类型;正确做法是先判断 Kind 是否支持,对 interface 类型需先用 Elem() 解包再判空。
reflect.ValueOf(i).IsNil() 会 panic这是最常踩的坑:对任意 interface{} 直接调用 IsNil(),Go 会 panic —— 因为 IsNil() 只允许作用于指针、切片、map、channel、func、interface 这六种类型;而 reflect.ValueOf(i) 返回的是一个 interface{} 的包装值,其底层 Kind 往往是 interface,但它的动态值(data)可能指向 int、string 等值类型,此时调用 IsNil() 就非法。
var i interface{} = 42
reflect.ValueOf(i).IsNil() // panic: call of reflect.Value.IsNil on int ValueKind 是否支持 IsNil(),再调用Value,再检查reflect.ValueOf(i).Kind() == reflect.Interface 后要再取 .Elem()
当 i 是一个非空接口(比如 io.Reader 或自定义接口),reflect.ValueOf(i) 的 Kind 是 interface,但它的值其实是“另一个 Value”——即接口的动态值。必须用 .Elem() 才能拿到那个实际值,否则永远在判断“接口头是否 nil”,而不是“它装的东西是否 nil”。
var r io.Reader = nil
v := reflect.ValueOf(r) // v.Kind() == reflect.Interface
if v.Kind() == reflect.Interface {
if !v.IsNil() { // 注意:这里 IsNil() 判的是 interface 头本身
v = v.Elem() // 必须 Elem() 才能访问底层值
}
}v.IsNil() 为 true,说明该接口变量本身是 nil(type==nil && data==nil),无需 Elem()
v.IsNil() 为 false,但 v.Elem().Kind() 是指针/map/slice 等,才可继续用 .IsNil()
nil 接口 + nil 指针/引用值生产环境里,你要处理的不是“理论上的空接口”,而是用户传进来的 interface{} 参数,它可能是 nil、可能是 *T、可能是 []int、也可能是 int。下面这个函数覆盖了常见情况:
func IsInterfaceNil(v interface{}) bool {
if v == nil {
return true
}
rv := reflect.ValueOf(v)
switch rv.Kind() {
case reflect.Chan, reflect.Func, reflect.Map, reflect.Ptr, reflect.Slice, reflect.UnsafePointer:
return rv.IsNil()
case reflect.Interface:
// 接口本身不为 nil,但内部值可能为 nil
if rv.IsNil() {

return true
}
// 解包后递归检查(避免无限递归,只解一层)
inner := rv.Elem()
if inner.IsValid() {
return IsInterfaceNil(inner.Interface())
}
return true
}
return false
}
v == nil 快速路径,开销最小reflect.Interface 类型,先 rv.IsNil() 判断接口头是否为空;不为空则 rv.Elem() 取内部值再判struct、int 等值类型——它们不可能是 nil,返回 false 合理interface{} → interface{} → *T)导致栈溢出反射在 Go 里是运行时开销大户:reflect.ValueOf() 分配堆内存,IsNil() 做类型检查和指针解引用。如果你的函数每秒被调用上万次,且多数输入是简单值类型(如 int、string),反射就成了瓶颈。
switch x := v.(type) {
case nil:
return true
case *T, []T, map[K]V, chan T, func():
return x == nil
default:
return false
}io.Reader),直接 v == nil 即可,根本不用反射interface{} 且无法预知内部结构**时,才用上面的反射方案v == nil),也可能是接口头非空但装着一个 nil *T,还可能是装着一个 nil map[string]int。反射只是工具,关键是你得清楚自己到底想捕获哪一种“空”。