首先通过反射获取源和目标对象的值,然后解引用指针并验证是否为结构体,接着遍历字段并复制可设置的字段值,最终实现通用深拷贝功能。

在 Golang 中,没有内置的深拷贝函数,但通过反射(reflect 包)可以实现一个通用的对象复制逻辑。这种能力在处理配置克隆、缓存对象或避免副作用时非常有用。本文介绍如何使用反射编写一个安全、可复用的拷贝函数,并探讨类型映射中的关键技巧。
Go 的 reflect 包允许程序在运行时检查变量的类型和值。核心是 reflect.Value 和 reflect.Type 两个类型。要实现拷贝,关键是能读取源对象字段并写入目标对象。
基本步骤包括:
下面是一个简单的结构体字段级拷贝示例:
立即学习“go语言免费学习笔记(深入)”;
func Copy(dst, src interface{}) error { dstV := reflect.ValueOf(dst) if dstV.Kind() != reflect.Ptr || dstV.IsNil() { return fmt.Errorf("dst must be a non-nil pointer") } srcV := reflect.ValueOf(src) if srcV.Kind() == reflect.Ptr { srcV = srcV.Elem() } dstV = dstV.Elem() if dstV.Kind() != reflect.Struct || srcV.Kind() != reflect.Struct { return fmt.Errorf("both src and dst must be structs") } t := srcV.Type() for i := 0; i这个函数支持结构体字段按名称匹配赋值,适用于大多数浅拷贝场景。
当结构体包含指针、slice 或其他复杂类型时,直接 Set() 只会复制引用。要实现深拷贝,需递归处理这些类型。
例如对 slice 字段:
可通过类型判断扩展拷贝行为:
if field.Kind() == reflect.Slice { newSlice := reflect.MakeSlice(field.Type(), field.Len(), field.Cap()) for j := 0; j实际项目中,两个结构体可能字段名相同但类型不同,或存在别名类型。此时需要更灵活的映射策略。
常见技巧包括:
例如:
type Source struct { ID int `copy:"uid"` Name string `copy:"username"` }解析 tag 后可实现跨命名拷贝。
基本上就这些。通过合理使用反射,Golang 能实现足够通用的拷贝功能。虽然性能不如手动赋值,但在需要泛化处理的场景下非常实用。注意边界情况:私有字段、不兼容类型、循环引用等都需要额外防护。不复杂但容易忽略。
以上就是如何用 Golang 反射实现通用的拷贝函数_Golang 对象复制与类型映射技巧的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号