【go】slice的浅拷贝和深拷贝

浅拷贝(Shallow Copy)

浅拷贝是指只复制切片本身的结构(指针、长度和容量),而不复制底层数组的元素。

实现方式

  1. 直接赋值

    go 复制代码
    slice1 := []int{1, 2, 3}
    slice2 := slice1  // 浅拷贝
  2. 切片操作

    go 复制代码
    slice1 := []int{1, 2, 3}
    slice2 := slice1[:]  // 浅拷贝

特点

  • 新切片和原切片共享同一个底层数组
  • 修改一个切片的元素会影响另一个切片
  • 内存效率高,因为不需要复制底层数据

示例

go 复制代码
package main

import "fmt"

func main() {
    original := []int{1, 2, 3}
    copy := original  // 浅拷贝
    
    copy[0] = 100
    
    fmt.Println(original)  // [100 2 3]
    fmt.Println(copy)      // [100 2 3]
}

深拷贝(Deep Copy)

深拷贝是指创建一个新的切片,并且复制底层数组的所有元素。

实现方式

  1. 使用copy函数

    go 复制代码
    slice1 := []int{1, 2, 3}
    slice2 := make([]int, len(slice1))
    copy(slice2, slice1)  // 深拷贝
  2. 使用append(需要先创建一个空切片):

    go 复制代码
    slice1 := []int{1, 2, 3}
    slice2 := append([]int{}, slice1...)  // 深拷贝

特点

  • 新切片有自己的底层数组
  • 修改一个切片的元素不会影响另一个切片
  • 内存消耗更大,因为需要复制所有数据

示例

go 复制代码
package main

import "fmt"

func main() {
    original := []int{1, 2, 3}
    copy := make([]int, len(original))
    copy(copy, original)  // 深拷贝
    
    copy[0] = 100
    
    fmt.Println(original)  // [1 2 3]
    fmt.Println(copy)      // [100 2 3]
}

注意事项

  1. copy函数的行为

    • 如果目标切片长度小于源切片,只会复制目标切片长度的元素
    • 如果目标切片长度大于源切片,多余的元素保持不变
  2. 多维切片的拷贝

    • 对于多维切片,copy函数或append只能实现第一层的深拷贝
    • 内层切片仍然是浅拷贝,需要递归处理才能实现完全深拷贝
  3. 性能考虑

    • 浅拷贝更快且内存效率更高
    • 深拷贝更安全但消耗更多资源

总结

特性 浅拷贝 深拷贝
实现方式 直接赋值或切片操作 copy函数或append
底层数组 共享 独立
修改影响 互相影响 互不影响
内存消耗
适用场景 不需要独立修改或关注性能的场景 需要独立修改或确保数据安全的场景
相关推荐
橙序员小站4 小时前
Agent Skill 是什么?一文讲透 Agent Skill 的设计与实现
前端·后端
怒放吧德德4 小时前
Netty 4.2 入门指南:从概念到第一个程序
java·后端·netty
雨中飘荡的记忆6 小时前
大流量下库存扣减的数据库瓶颈:Redis分片缓存解决方案
java·redis·后端
开心就好20257 小时前
UniApp开发应用多平台上架全流程:H5小程序iOS和Android
后端·ios
悟空码字7 小时前
告别“屎山代码”:AI 代码整洁器让老项目重获新生
后端·aigc·ai编程
小码哥_常8 小时前
大厂不宠@Transactional,背后藏着啥秘密?
后端
奋斗小强8 小时前
内存危机突围战:从原理辨析到线上实战,彻底搞懂 OOM 与内存泄漏
后端
小码哥_常8 小时前
Spring Boot接口防抖秘籍:告别“手抖”,守护数据一致性
后端
心之语歌8 小时前
基于注解+拦截器的API动态路由实现方案
java·后端
None3218 小时前
【NestJs】基于Redlock装饰器分布式锁设计与实现
后端·node.js