用 reflect.TypeOf(v).Kind() == reflect.Slice 判断是否为 slice,需先用 reflect.ValueOf(v) 处理接口和 nil,再循环解指针;数组 [N]T 的 Kind 是 Array 而非 Slice,不可混淆。
Go 中没有直接的 isSlice() 函数,必须借助 reflect 包。关键不是看变量名或声明,而是看运行时类型;reflect.TypeOf(v).Kind() 返回的是底层类型分类,对所有切片(无论元素类型)都返回 reflect.Slice。
.Kind() 会是 Ptr、Struct、Map,不是 Slice
nil 接口直接调用 reflect.TypeOf(),会 panic;需先判空或用 reflect.ValueOf() 配合 .IsValid()
reflect.TypeOf([]int{}).Kind() == reflect.Slice ✅reflect.TypeOf([3]int{}).Kind() == reflect.Array ❌(数组不是 slice)很多人混淆 []T 和 [N]T,但它们在反射中完全不同的 Kind。仅靠 .Kind() == reflect.Slice 才是真正 slice;若想进一步确认元素类型,可用 .Elem() 获取切片的元素类型。
reflect.Slice:对应 []string、[]*http.Request 等动态长度切片reflect.Array:对应 [5]int、[0]byte 等固定长度数组reflect.TypeOf([2]int{}).Kind() == reflect.Slice → 实际返回 Array,结果为 false下面是一个安全、泛用的 IsSlice() 函数,能处理接口、nil、指针指向 slice 等边界情况:
func IsSlice(v interface{}) bool {
rv := reflect.ValueOf(v)
if !rv.IsValid() {
return false
}
// 解引用指针,例如 *[]int → []int
for rv.Kind() == reflect.Ptr {
rv = rv.Elem()
if !rv.IsValid() {
return false
}
}
return rv.Kind() == reflect.Slice
}
IsSlice([]string{}) → trueIsSlice(&[]byte{}) → true(自动解指针)IsSlice(nil) → false(不会 panic)IsSlice(42) 或 IsSlice("hello") → false有人尝试用字符串匹配,比如检查 reflect.ValueOf(x).Type().String() 是否包含 "[]",这是不可靠的。因为:
type MySlice []int 的 .String() 是 "m
ain.MySlice",不含 "[]"
[][]string 的字符串是 "[][]string",正则易误判.Kind() 是反射系统定义的稳定枚举,语义明确,性能也更好.Kind() 判断大类(slice/array/map/chan),再用 .Elem() 或 .In() 查细节类型系统里,“是不是 slice”是个 Kind 层级问题,不是字符串格式问题。