Go语言通过reflect包实现反射,可动态获取变量的类型(reflect.Type)和值(reflect.Value)。利用TypeOf和ValueOf能处理任意类型数据,适用于通用函数、序列化等场景。通过Kind()方法判断底层类型(如Ptr、Slice),避免冗余的类型断言。反射还支持遍历结构体字段并解析标签(如json标签),常用于ORM、JSON序列化等,仅能访问导出字段(首字母大写)。

Go语言的
reflect
在
reflect
reflect.Type
reflect.Value
通过
reflect.TypeOf()
reflect.ValueOf()
reflect.TypeOf(x)
reflect.Type
reflect.ValueOf(x)
reflect.Value
示例:
立即学习“go语言免费学习笔记(深入)”;
var name string = "Tom" t := reflect.TypeOf(name) // string v := reflect.ValueOf(name) // "Tom" fmt.Println(t, v)
反射可以避免使用大量
switch v.(type)
利用
Kind()
struct
slice
ptr
reflect.Type.Kind()
*int
Kind()
ptr
[]string
slice
示例:检查是否为指针类型
func checkType(v interface{}) {
rv := reflect.ValueOf(v)
if rv.Kind() == reflect.Ptr {
fmt.Println("这是一个指针类型")
} else {
fmt.Println("不是指针类型")
}
}
checkType(&name) // 输出:这是一个指针类型
反射常用于解析结构体字段及其标签,比如JSON序列化、ORM映射等场景。
通过
reflect.Type.Field(i)
Field(i)
StructField
Name
Type
Tag
Tag.Get("json")示例:读取结构体的JSON标签
type User struct {
Name string `json:"user_name"`
Age int `json:"user_age"`
}
u := User{Name: "Alice", Age: 25}
t := reflect.TypeOf(u)
for i := 0; i < t.NumField(); i++ {
field := t.Field(i)
jsonTag := field.Tag.Get("json")
fmt.Printf("字段 %s 对应 JSON key: %s\n", field.Name, jsonTag)
}
// 输出:
// 字段 Name 对应 JSON key: user_name
// 字段 Age 对应 JSON key: user_age
反射不仅能读取值,还能修改值,但必须确保值是“可寻址”且“可设置”的。
直接传值调用
reflect.ValueOf(v)
Value
.Elem()
reflect.Value
.CanSet()
示例:通过反射修改字符串值
func setString(v interface{}, newVal string) {
rv := reflect.ValueOf(v)
if rv.Kind() == reflect.Ptr && !rv.Elem().CanSet() {
fmt.Println("无法设置该值")
return
}
elem := rv.Elem()
if elem.Kind() == reflect.String {
elem.SetString(newVal)
}
}
var s string = "old"
setString(&s, "new")
fmt.Println(s) // 输出: new
基本上就这些。Go的反射机制虽然强大,但使用时需注意性能开销和安全性。动态类型检查在泛型缺失时期尤为重要,随着Go 1.18+泛型的引入,部分场景可用更安全高效的泛型替代。但在元编程、配置解析、RPC框架等场景中,
reflect
以上就是Golang的reflect反射机制 动态类型检查的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号