你以为传切片就是传引用了吗?

简介: xdm ,我们在写 golang 的时候,引用和传值傻傻分不清,就例如我们传 切片 的时候,你能分清楚你传的切片是传值还是传引用呢?

xdm ,我们在写 golang 的时候,引用和传值傻傻分不清,就例如我们传 切片 的时候,你能分清楚你传的切片是传值还是传引用呢?

引用是什么?

引用就是给对象起另一个名字,引用类型引用另一种类型

引用并不是对象,相反的,它只是为一个已存在的对象所起的另外一个名字

我们在写 C++ 的时候,知道引用其实就是实际值的另外一个别名,有的会举例子说是一个门牌号的概念

你以为的 引用传递

写个例子,咱们创建一个 切片 s1,

  • s1 切片赋值 , 1, 2, 3, 4, 5 , 输出效果
  • 将 s1 传递给 myModify ,在函数中修改 第 3 个元素第 4 个元素的值
func main() {
  s1 := make([]int, 5)
  s1 = []int{ 1, 2, 3, 4, 5}
  fmt.Println("s1 = ", s1)
  myModify(s1)
  fmt.Println("myAdd result = ", s1)
}
func myModify(ss []int ) {
  ss[2] = 8
  ss[3] = 9
}

查看效果

# go run main.go
s1 =  [1 2 3 4 5]
myAdd result =  [1 2 8 9 5]

看到这里,你以为的引用传递是不是就是这个样子的,因为传入到函数中的切片,在函数中把传进来的切片的值修改,外面实际的切片对应的值也改变了

xdm , 看到这里你就下定论是不是不太好,咱们再来探究一下

确认还是引用吗?

咱们在上面的实现上面具体的打印出 切片的地址来看看效果,我们就稍微有点感觉了

func main() {
  s1 := make([]int, 5)
  s1 = []int{ 1, 2, 3, 4, 5}
  fmt.Printf("s1地址  =  %p , s1: %v \r\n",&s1, s1)
  myModify(s1)
  fmt.Printf("s1地址  =  %p , s1: %v \r\n",&s1, s1)
}
func myModify(ss []int ) {
  ss[2] = 8
  ss[3] = 9
  fmt.Printf("ss地址  =  %p , s1: %v \r\n",&ss, ss)
}


运行程序,看实际效果

# go run main.go
s1地址  =  0xc42000a060 , s1: [1 2 3 4 5]
ss地址  =  0xc42000a0a0 , s1: [1 2 8 9 5]
s1地址  =  0xc42000a060 , s1: [1 2 8 9 5]

这个时候,你还觉得咱们传递的是引用吗?这个做法还是引用传递吗?咱们要好好思考一下

s1切片的地址 是 0xc42000a060 ,ss 切片的地址是 0xc42000a0a0,这里很明显看出来 s1  和 ss 并不是同一个东西,因此,我们可以得出结论,切片作为函数参数,是值传递

可以为什么 s1 和 ss 打印出来的数据是一样的呢,而且修改了 ss 切片里面的元素为什么会影响到原来 s1 的切片呢?

开始探究

GO 里面的切面的底层是是一个结构体

type SliceHeader struct {
  Data uintptr
  Len  int
  Cap  int
}

对于上面的问题, s1 和 ss 变量,有各自变量的内存地址,但是这个变量下面对应的底层数组是对应的同一片空间,因此 s1 的变动会影响到 ss 切片的变动

对于这个我们也是很好验证的,来写一个例子,将上述的两个切片里面的每隔元素的地址都打印出来对比一下:

func main() {
  s1 := make([]int, 5)
  s1 = []int{1, 2, 3, 4, 5}
  fmt.Printf("s1地址  =  %p , s1: %v \r\n", &s1, s1)
  for k, v := range s1 {
    fmt.Printf("s1[%d]地址  =  %p , s1: %v \r\n", k, &s1[k], v)
  }
  myModify(s1)
  fmt.Printf("s1地址  =  %p , s1: %v \r\n", &s1, s1)
}
func myModify(ss []int) {
  ss[2] = 8
  ss[3] = 9
  fmt.Printf("ss地址  =  %p , s1: %v \r\n", &ss, ss)
  for k, v := range ss {
    fmt.Printf("ss[%d]地址  =  %p , s1: %v \r\n", k, &ss[k], v)
  }
}

查看效果

image.png

根据上述效果,我们可以看到,s1 和 ss ,只是自己变量的地址不同,但是自己指向的底层数组的内存完全相同

平时工作中,这里就需要注意了,你以为的你以为真不可能不是这样的!!


欢迎点赞,关注,收藏

朋友们,你的支持和鼓励,是我坚持分享,提高质量的动力

image.png

好了,本次就到这里

技术是开放的,我们的心态,更应是开放的。拥抱变化,向阳而生,努力向前行。

我是阿兵云原生,欢迎点赞关注收藏,下次见~

相关文章
|
1天前
|
弹性计算 关系型数据库 微服务
基于 Docker 与 Kubernetes(K3s)的微服务:阿里云生产环境扩容实践
在微服务架构中,如何实现“稳定扩容”与“成本可控”是企业面临的核心挑战。本文结合 Python FastAPI 微服务实战,详解如何基于阿里云基础设施,利用 Docker 封装服务、K3s 实现容器编排,构建生产级微服务架构。内容涵盖容器构建、集群部署、自动扩缩容、可观测性等关键环节,适配阿里云资源特性与服务生态,助力企业打造低成本、高可靠、易扩展的微服务解决方案。
1063 0
|
10天前
|
人工智能 运维 安全
|
1天前
|
弹性计算 Kubernetes jenkins
如何在 ECS/EKS 集群中有效使用 Jenkins
本文探讨了如何将 Jenkins 与 AWS ECS 和 EKS 集群集成,以构建高效、灵活且具备自动扩缩容能力的 CI/CD 流水线,提升软件交付效率并优化资源成本。
248 0
|
8天前
|
人工智能 异构计算
敬请锁定《C位面对面》,洞察通用计算如何在AI时代持续赋能企业创新,助力业务发展!
敬请锁定《C位面对面》,洞察通用计算如何在AI时代持续赋能企业创新,助力业务发展!
|
9天前
|
人工智能 测试技术 API
智能体(AI Agent)搭建全攻略:从概念到实践的终极指南
在人工智能浪潮中,智能体(AI Agent)正成为变革性技术。它们具备自主决策、环境感知、任务执行等能力,广泛应用于日常任务与商业流程。本文详解智能体概念、架构及七步搭建指南,助你打造专属智能体,迎接智能自动化新时代。
|
9天前
|
机器学习/深度学习 人工智能 自然语言处理
B站开源IndexTTS2,用极致表现力颠覆听觉体验
在语音合成技术不断演进的背景下,早期版本的IndexTTS虽然在多场景应用中展现出良好的表现,但在情感表达的细腻度与时长控制的精准性方面仍存在提升空间。为了解决这些问题,并进一步推动零样本语音合成在实际场景中的落地能力,B站语音团队对模型架构与训练策略进行了深度优化,推出了全新一代语音合成模型——IndexTTS2 。
740 23