【golang】深入理解切片(slice)函数传递

问题

  1. 第一段代码
go 复制代码
func main() {
	s := make([]int, 0, 5)
	s = append(s, 1)
	fmt.Println("append slice:", s, len(s), cap(s))
}

这段代码中为长度为0,容量为5的切片append了一个1,切片未发生扩容,所以代码运行结果为切片第一个元素为1,长度为1,容量为5的切片。

运行结果:

go 复制代码
append slice: [1] 1 5
  1. 第二段代码
go 复制代码
func main() {
	s := make([]int, 0, 5)
	s = append(s, 1)
	fmt.Println("append slice:", s, len(s), cap(s))
	appendFunc(s)
	fmt.Println("appendFuncMain slice:", s, len(s), cap(s))
}

func appendFunc(s []int) {
	s = append(s, 1, 2, 3)
}

这段代码添加了一个appendFunc函数,函数功能为切片增加1,2,3三个元素。在主函数中调用了appendFunc函数并把切片s作为参数传递。

运行结果:

go 复制代码
append slice: [1] 1 5
appendFuncMain slice: [1] 1 5

为什么切片s的长度和值都没有变呢?

  1. 第三段代码
go 复制代码
func main() {
	s := make([]int, 0, 5)
	s = append(s, 1)
	fmt.Println("append slice:", s, len(s), cap(s))
	appendFunc(s)
	fmt.Println("appendFuncMain slice:", s, len(s), cap(s))

}

func appendFunc(s []int) {
	s = append(s, 1, 2, 3)
	fmt.Println("appendFunc slice:", s, len(s), cap(s))
}

这段代码新加了一个输出在appendFunc函数中,我们来看运行结果:

go 复制代码
append slice: [1] 1 5
appendFunc slice: [1 1 2 3] 4 5
appendFuncMain slice: [1] 1 5 

可以看到,在appendFunc函数中打印传递的切片s是正常变化了的,这是怎么回事呢?切片不是引用类型吗?

解答

切片在go语言中属于引用类型,但是在go中都是值传递(拷贝副本),那切片是传递的什么呢?

切片的底层其实是一个结构体类型,在src/runtime/slice.go下可以看到。

go 复制代码
type slice struct {
	array unsafe.Pointer
	len   int
	cap   int
}

array是指向底层数组,len为切片长度,cap为切片容量。在函数传递中就是把这个结构体拷贝了一份副本进行传递。

所以上面问题就得到了解答,因为拷贝的是slice结构体的副本,而append实际上是增加了副本的len,而传递的原slice结构体是没有发生改变的。

我们可以加个指针继续验证一下:

go 复制代码
func main() {
	s := make([]int, 0, 5)
	s = append(s, 1)
	fmt.Println("append slice:", s, len(s), cap(s))
	appendFunc(&s)
	fmt.Println("appendFuncMain slice:", s, len(s), cap(s))

}

func appendFunc(s *[]int) {
	*s = append(*s, 1, 2, 3, 5, 6, 7, 5)
	fmt.Println("appendFunc slice:", s, len(*s), cap(*s))
}

输出结果:

go 复制代码
append slice: [1] 1 5
appendFunc slice: &[1 1 2 3 5 6 7 5] 8 10   
appendFuncMain slice: [1 1 2 3 5 6 7 5] 8 10

加上指针后,函数传递就是slice切片指针地址值,作append增加或修改操作都是对一个slice 对象修改的,所以在不同作用域下修改的对象都是一致的。

相关推荐
Mr -老鬼5 分钟前
Rust适合干什么?为什么需要Rust?
开发语言·后端·rust
予枫的编程笔记8 分钟前
【Java集合】深入浅出 Java HashMap:从链表到红黑树的“进化”之路
java·开发语言·数据结构·人工智能·链表·哈希算法
ohoy14 分钟前
RedisTemplate 使用之Set
java·开发语言·redis
mjhcsp14 分钟前
C++ 后缀数组(SA):原理、实现与应用全解析
java·开发语言·c++·后缀数组sa
hui函数15 分钟前
如何解决 pip install 编译报错 ‘cl.exe’ not found(缺少 VS C++ 工具集)问题
开发语言·c++·pip
西京刀客19 分钟前
golang路由与框架选型(对比原生net/http、httprouter、Gin)
http·golang·gin
云栖梦泽24 分钟前
易语言Windows桌面端「本地AI知识管理+办公文件批量自动化处理」双核心系统
开发语言
Mr -老鬼27 分钟前
Rust与Go:从学习到实战的全方位对比
学习·golang·rust
r_oo_ki_e_34 分钟前
java22--常用类
java·开发语言
AI小怪兽36 分钟前
轻量、实时、高精度!MIE-YOLO:面向精准农业的多尺度杂草检测新框架 | MDPI AgriEngineering 2026
开发语言·人工智能·深度学习·yolo·无人机