使用reflect实现通用拷贝需确保源和目标类型一致且目标为可设置的指针;2. 通过Elem()获取指针指向的值;3. 遍历结构体字段或递归处理嵌套类型;4. 仅复制公共字段(首字母大写);5. 支持基础类型、切片、map及指针的深拷贝,但不处理循环引用。

在Go语言中,反射(reflect)可以用来实现通用的数据拷贝函数,适用于结构体、切片、map等复杂类型。通过reflect包,我们可以在运行时动态获取变量的类型和值,并进行字段级别的复制,特别适合用于深拷贝场景。
基本思路与关键点
使用反射实现通用拷贝,核心是判断源对象和目标对象的类型是否匹配,然后逐字段赋值。目标通常需要是指针,以便修改其内容。
- 源对象和目标对象必须是相同的可导出结构体类型,或兼容的引用类型
- 只拷贝公共字段(首字母大写)
- 支持嵌套结构体、基础类型、指针、slice、map等
- 避免循环引用(本示例不处理)
通用拷贝函数实现
func Copy(dst, src interface{}) error { dstVal := reflect.ValueOf(dst) srcVal := reflect.ValueOf(src) if dstVal.Kind() != reflect.Ptr || dstVal.IsNil() { return fmt.Errorf("dst must be a non-nil pointer") } if srcVal.Kind() == reflect.Ptr { srcVal = srcVal.Elem() } dstVal = dstVal.Elem() if !dstVal.CanSet() { return fmt.Errorf("dst is not settable") } if dstVal.Type() != srcVal.Type() { return fmt.Errorf("src and dst must have the same type") } copyRecursive(dstVal, srcVal) return nil } func copyRecursive(dst, src reflect.Value) { switch src.Kind() { case reflect.Struct: for i := 0; i使用示例
定义一个包含嵌套结构体、slice和map的结构体,测试拷贝功能:
type Address struct { City string Zip string } type Person struct { Name string Age int Addr *Address Emails []string Metadata map[string]interface{} } // 示例调用 func main() { src := Person{ Name: "Alice", Age: 30, Addr: &Address{City: "Beijing", Zip: "100000"}, Emails: []string{"a@1.com", "a@2.com"}, Metadata: map[string]interface{}{ "job": "engineer", "tags": []string{"golang", "dev"}, }, } var dst Person err := Copy(&dst, &src) if err != nil { log.Fatal(err) } // 修改源数据,验证是否深拷贝 src.Emails[0] = "changed@1.com" src.Metadata["job"] = "manager" fmt.Printf("Dst.Emails[0]: %s\n", dst.Emails[0]) // 不受影响 fmt.Printf("Dst.Metadata[\"job\"]: %s\n", dst.Metadata["job"]) // 不受影响 }输出结果表明,dst中的数据独立于src,说明实现了深拷贝。
立即学习“go语言免费学习笔记(深入)”;
基本上就这些。这个通用拷贝函数虽然不能处理所有极端情况(如channel、func、复杂循环引用),但在大多数结构体数据复制场景中已经足够实用。










