登录
首页 >  Golang >  Go教程

结构体含切片map复制是浅拷贝

时间:2025-09-24 11:38:26 391浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《结构体含切片或map复制会浅拷贝》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下,希望所有认真读完的童鞋们,都有实质性的提高。

结构体值复制时,切片和map字段共享底层数据,仅复制引用;修改元素会影响对方,append可能触发扩容导致分离;map修改则双方均可见;需手动深拷贝实现完全独立。

当Golang结构体包含切片或map时作为值类型复制会发生什么

当 Go 语言中的结构体包含切片(slice)或映射(map)时,如果该结构体作为值类型进行复制(例如赋值、传参或返回),结构体本身会被完整复制,但其中的切片和 map 字段并不会被深度复制。

结构体值复制时的行为

Go 中的结构体是值类型。当你将一个结构体变量赋值给另一个变量,或将其作为参数传递给函数时,会创建该结构体的一个副本。这个副本包含原始结构体所有字段的拷贝。

对于基本类型字段(如 int、string、bool 等),它们的值会被直接复制。但对于引用类型字段,如 slice、map 和指针,复制的是它们的“引用”或“头信息”,而不是底层数据。

切片字段的复制

切片本质上是一个指向底层数组的指针(包含长度和容量信息的结构体)。当结构体被复制时,切片字段的指针部分也被复制,因此两个结构体的切片字段会指向同一个底层数组。

这意味着:
  • 修改一个结构体中切片的元素会影响另一个结构体中对应切片的内容(只要没有触发扩容)
  • 对切片执行 append 操作可能导致扩容,从而让两个切片分离

示例:

type Data struct {
    Items []int
}

d1 := Data{Items: []int{1, 2, 3}}
d2 := d1  // 值复制

d1.Items[0] = 99
fmt.Println(d2.Items) // 输出 [99 2 3],因为共享底层数组

d1.Items = append(d1.Items, 4) // 可能触发扩容
// 此时 d1.Items 和 d2.Items 可能不再共享底层数组

Map 字段的复制

map 是引用类型。结构体复制时,map 字段的引用被复制,两个结构体的 map 字段指向同一个底层 map。

这意味着:
  • 通过任一结构体修改 map 中的键值,另一个结构体也能看到变化
  • 删除或添加键值对会影响共享的 map

示例:

type Info struct {
    Props map[string]int
}

i1 := Info{Props: map[string]int{"a": 1}}
i2 := i1  // 值复制

i1.Props["a"] = 99
fmt.Println(i2.Props["a"]) // 输出 99

i1.Props["b"] = 2
fmt.Println(i2.Props["b"]) // 输出 2

如何避免共享数据

如果你希望两个结构体完全独立,需要手动进行深度复制:

  • 对切片:使用 make + copy,或遍历赋值
  • 对 map:创建新 map 并逐个复制键值对

手动深拷贝示例:

d2 := Data{
    Items: make([]int, len(d1.Items)),
}
copy(d2.Items, d1.Items)

i2 := Info{
    Props: make(map[string]int),
}
for k, v := range i1.Props {
    i2.Props[k] = v
}

基本上就这些。结构体值复制时,切片和 map 字段共享底层数据,行为取决于是否修改了这些引用类型的内容。理解这一点对避免意外的副作用很重要。

好了,本文到此结束,带大家了解了《结构体含切片map复制是浅拷贝》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>