正确获取切片/数组长度需先用Kind()判断类型,对指针解引用再调Len();空切片Len()安全返回0,Index()需手动边界检查;遍历前确认字段导出、非nil且为struct类型;避免循环内重复ValueOf。
reflect.ValueOf 正确获取切片/数组长度和元素反射访问切片或数组前,必须先确认值是否为可寻址且非 nil。直接对 nil 切片调用 Len() 会 panic;对指针类型未解引用也会导致 Len() 返回 0 或 panic。
v.Kind() == reflect.Slice || v.Kind() == reflect.Array 做类型判断,别只靠 Interface() 断言*[]int),需先调用 v.Elem() 解引用,否则 Len() 不可用[]int(nil))调用 v.Len() 是安全的,返回 0;但 v.Index(0) 会 panicreflect.Value.Index(i) 遍历时的边界与 panic 风险反射索引访问不自动做越界检查——它只在 i >= v.Len() 时 panic,但不会检查负数索引是否合法(负数直接 panic)。这和原生切片行为一致,但容易在动态计算下出错。
v.Len() 获取真实长度,不要依赖 v.Cap()(对数组无效,对切片可能大于长度)for i := 0
; i —— 多一次迭代必然 panic
func safeIndex(v reflect.Value, i int) (reflect.Value, bool) {
if i < 0 || i >= v.Len() {
return reflect.Value{}, false
}
return v.Index(i), true
}reflect.Value 类型链容易断裂当从结构体字段取到一个切片字段(如 user.Orders []Order),再对其元素做反射操作时,每一步都可能返回不可寻址或不可设置的 Value,尤其在字段是 unexported(小写开头)时。
v.FieldByName("Orders") 返回零值,Len() 为 0 且无法继续*[]string),此时要先 v.FieldByName("Orders").Elem() 再判空FieldByName 前,确保该元素是 struct 类型:用 elem.Kind() == reflect.Struct 先过滤reflect.TypeOf 或 reflect.ValueOf
反射开销集中在类型检查与动态调度上。reflect.ValueOf(x) 在循环中反复调用,等于每次重新包装接口,比复用已有 reflect.Value 慢 3–5 倍(实测 Go 1.21)。
reflect.ValueOf(slice) 提到循环外;若需多次遍历,复用该 Value 实例reflect.Value.Interface() 转回 interface{} 再断言——这会触发额外内存分配unsafe + reflect.SliceHeader 替代(仅限已知底层数组且无 GC 压力时)