在Go语言中,当你需要在不同的变量之间传递数据时,你可能会遇到深拷贝与浅拷贝的概念。浅拷贝是指创建一个新的对象引用,但是该引用仍然指向原有对象中的数据;深拷贝则是创建一个新的对象,并复制原有对象中的数据,新旧对象互不影响。
浅拷贝
在Go中,当你将一个对象赋值给另一个对象时,如果这个对象是一个指针类型,那么它就是浅拷贝。
type Person struct {
    Name string
    Age  int
}
 
func shallowCopy(p Person) Person {
    return p
}
 
func main() {
    original := Person{"Alice", 30}
    copy := shallowCopy(original)
    copy.Name = "Bob"
    fmt.Println(original.Name) // 输出 Bob
}
在这个例子中,original 和 copy 指向的是同一个 Person 结构体实例。因此,改变 copy.Name 也会改变 original.Name。
深拷贝
为了进行深拷贝,你需要创建一个新的对象,并复制原对象的数据。在Go中,你可以使用 copy 包的 Copy 函数来实现深拷贝。
func deepCopy(p Person) Person {
    var p2 Person
    bytes.Buffer.ReadFrom(&p)
    if err := gob.NewDecoder(&buf).Decode(&p2); err != nil {
        log.Fatal(err)
    }
    return p2
}
 
func main() {
    original := Person{"Alice", 30}
    copy := deepCopy(original)
    copy.Name = "Bob"
    fmt.Println(original.Name) // 输出 Alice
}
在这个例子中,deepCopy 函数使用了 encoding/gob 包来进行深拷贝。这种方法虽然可以实现深拷贝,但是有一定的限制,例如,拷贝的对象需要是 gob 支持的类型。
自定义深拷贝
对于不希望使用 gob 的情况,你可以自定义深拷贝的逻辑。
func deepCopy(p Person) Person {
    return Person{Name: string([]byte(p.Name)), Age: p.Age}
}
 
func main() {
    original := Person{"Alice", 30}
    copy := deepCopy(original)
    copy.Name = "Bob"
    fmt.Println(original.Name) // 输出 Alice
}
在这个例子中,deepCopy 函数通过创建新的 Name 字段,从而实现深拷贝。这样,改变 copy.Name 不会影响 original.Name。
总结,深拷贝和浅拷贝的主要区别在于它们如何处理对象中的数据。浅拷贝只是复制了引用,而深拷贝则创建了新的数据副本。在Go中,你可以通过自定义复制逻辑或使用特定的库来实现深拷贝。