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

在 Golang 中,没有内置的深拷贝函数,但通过反射(reflect 包)可以实现一个通用的对象复制逻辑。这种能力在处理配置克隆、缓存对象或避免副作用时非常有用。本文介绍如何使用反射编写一个安全、可复用的拷贝函数,并探讨类型映射中的关键技巧。
理解反射的基本操作
Go 的 reflect 包允许程序在运行时检查变量的类型和值。核心是 reflect.Value 和 reflect.Type 两个类型。要实现拷贝,关键是能读取源对象字段并写入目标对象。
基本步骤包括:
- 通过 reflect.ValueOf(obj) 获取值反射对象
- 使用 .Elem() 解引用指针(如果是指针)
- 遍历结构体字段:用 .NumField() 和 .Field(i)
- 判断字段是否可设置:.CanSet()
实现基础拷贝函数
下面是一个简单的结构体字段级拷贝示例:
立即学习“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 字段:
- 创建新 slice,长度与原 slice 一致
- 逐个元素调用拷贝逻辑(如果是结构体)
- 对于 map 类型,需新建 map 并复制键值对
可通过类型判断扩展拷贝行为:
if field.Kind() == reflect.Slice { newSlice := reflect.MakeSlice(field.Type(), field.Len(), field.Cap()) for j := 0; j类型兼容性与字段映射技巧
实际项目中,两个结构体可能字段名相同但类型不同,或存在别名类型。此时需要更灵活的映射策略。
常见技巧包括:
- 使用 struct tag 标记映射规则,如 `copy:"name"`
- 在拷贝前做类型兼容判断:AssignableTo() 或 ConvertableTo()
- 对时间戳、自定义类型等特殊字段做单独处理
例如:
type Source struct { ID int `copy:"uid"` Name string `copy:"username"` }解析 tag 后可实现跨命名拷贝。
基本上就这些。通过合理使用反射,Golang 能实现足够通用的拷贝功能。虽然性能不如手动赋值,但在需要泛化处理的场景下非常实用。注意边界情况:私有字段、不兼容类型、循环引用等都需要额外防护。不复杂但容易忽略。










