- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
作者:林冠宏 / 指尖下的幽灵。转载者,请: 务必标明出处.
GitHub : https://github.com/af913337456/ 。
出版的书籍:
- 《1.0-区块链DApp开发实战》
- 《2.0-区块链DApp开发:基于公链》
切片 slice 几乎是每个 Go 开发者以及项目中 100% 会高频使用到的,Go 语言的知识很广,唯独 slice 我个人认为是必须要深入了解的.
乃至于今,网上还有很多关于切片 slice 技术文章一直存在的错误内容:切片作为函数参数传递的是引用,这是错误的.
无论是官方说明还是实践操作都表明:切片作为函数参数传递的是值,和数组一样.
接下来我们直接看例子以加深印象.
func main() {
mSlice := []int{1, 2, 3}
fmt.Printf("main-1: %p \n", &mSlice) // 0x140000b2000
mAppend(mSlice)
fmt.Printf("main-2: %p \n", &mSlice) // 0x140000b2000
}
func mAppend(slice []int) {
fmt.Printf("append func: %p \n", &slice) // 0x140000b2018 和外部的不一样
}
现在用来误导切片作为函数参数传递的是引用的错误文章常用的
:func main() {
mSlice := []int{1, 2, 3}
fmt.Printf("main-1: %v \n", mSlice) // [1,2,3]
mAppend(mSlice)
fmt.Printf("main-2: %v \n", mSlice) // [1,9,3],这里2被修改了,但不是引用传递导致的
}
func mAppend(slice []int) {
slice[2] = 9 // 修改
}
// 源码路径:go/src/runtime/slice.go
type slice struct {
array unsafe.Pointer // 指针
len int
cap int
}
切片的本质是 struct,作为函数参数传递时候遵循 struct 性质,array 是指针指向一个数组,len 是数组的元素个数,cap 是数组的的长度。当 len > cap,将触发数组扩容.
解析: 为什么上面的 错觉例子 能在函数内部改变值且在外部生效。这是因为当切片作为参数传递到函数里,虽然是值传递,但函数内拷贝出的新切片的 array 指针所指向的数组和外部的旧切片是一样的,那么在没引起扩容情况下进行值的修改就生效了.
旧切片 array 指针 ---> 数组-1 。
新切片 array 指针 ---> 数组-1,函数内发生改变 。
func main() {
mSlice := []int{1, 2, 3}
fmt.Printf("main-1: %v \n", mSlice) // [1,2,3]
mAppend(mSlice)
fmt.Printf("main-2: %v \n", mSlice) // [1,2,3] 没生效
}
func mAppend(slice []int) {
// slice[2] = 9 // 修改
slice = append(slice, 4)
fmt.Printf("append: %v \n", slice) // [1,2,3,4]
}
解析:切片初始化时候添加了3个数,导致其 len 和 cap 都是3,函数内添加第四个数的时候,触发扩容,而扩容会导致扩容,array 指针指向新的数组,在函数结束后,旧切片数组并没修改.
旧切片 array 指针 ---> 数组-1 值 [1,2,3] 。
新切片 array 指针 ---> 数组-2 值 [1,2,3,4] 。
func main() {
mSlice := make([]int, 3, 4) // len = 3, cap = 4, cap > len
fmt.Printf("main-1: %v, 数组地址: %p \n", mSlice, mSlice) // [0,0,0], 0x14000120000
mAppend(mSlice)
fmt.Printf("main-2: %v, 数组地址: %p \n", mSlice, mSlice) // [0,0,0], 0x14000120000
}
func mAppend(slice []int) {
slice = append(slice, 4)
fmt.Printf("append: %v, 数组地址: %p \n", slice, slice) // [0,0,0,4], 0x14000120000
}
解析:可以看到切片的底层数组地址并没改变,但是数组的值依然没改变。这是因为切片是值传递到函数内部的,此时的 len 依然是值传递,当打印的时候,就只打印 len 以内的数据.
旧切片 len = 3 。
新切片 len = 4,函数内改变 。
至此,我们应该如何让切片在函数内的修改生效?答案就是规规矩矩使用指针传参.
func main() {
mSlice := []int{1, 2, 3}
fmt.Printf("main-1: %v, 数组地址: %p \n", mSlice, mSlice) // [1,2,3], 0x1400001a0a8
mAppend(&mSlice)
fmt.Printf("main-2: %v, 数组地址: %p \n", mSlice, mSlice) // [1,2,3,4], 0x1400001a0a8
}
func mAppend(slice *[]int) {
*slice = append(*slice, 4)
fmt.Printf("append: %v, 数组地址: %p \n", *slice, slice) // [1,2,3,4], 0x140000181b0
}
上面例子成功在函数内使用 append 修改了切片,也可以看到切片数组地址变了,这是因为引起了扩容。但 array 指针没变,所以扩容后,指向了新的.
旧切片 array 指针 ---> 数组-1 值 [1,2,3] 。
旧切片 array 指针 ---> 数组-2 值 [1,2,3,4] 。
切片的扩容:
go1.18之前,临界值为1024,len 小于1024时,切片先2倍 len 扩容。大于 1024,每次增加 25% 的容量,直到新容量大于期望容量; 。
go1.18之后,临界值为256,len 小于256,依然2倍 len 扩容。大于256走算法:newcap += (newcap + 3*threshold) / 4,直到满足。(threshold = 256) 。
最后此篇关于Golang切片作为函数参数传递的陷阱与解答的文章就讲到这里了,如果你想了解更多关于Golang切片作为函数参数传递的陷阱与解答的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
C语言sscanf()函数:从字符串中读取指定格式的数据 头文件: ?
最近,我有一个关于工作预评估的问题,即使查询了每个功能的工作原理,我也不知道如何解决。这是一个伪代码。 下面是一个名为foo()的函数,该函数将被传递一个值并返回一个值。如果将以下值传递给foo函数,
CStr 函数 返回表达式,该表达式已被转换为 String 子类型的 Variant。 CStr(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CSng 函数 返回表达式,该表达式已被转换为 Single 子类型的 Variant。 CSng(expression) expression 参数是任意有效的表达式。 说明 通常,可
CreateObject 函数 创建并返回对 Automation 对象的引用。 CreateObject(servername.typename [, location]) 参数 serv
Cos 函数 返回某个角的余弦值。 Cos(number) number 参数可以是任何将某个角表示为弧度的有效数值表达式。 说明 Cos 函数取某个角并返回直角三角形两边的比值。此比值是
CLng 函数 返回表达式,此表达式已被转换为 Long 子类型的 Variant。 CLng(expression) expression 参数是任意有效的表达式。 说明 通常,您可以使
CInt 函数 返回表达式,此表达式已被转换为 Integer 子类型的 Variant。 CInt(expression) expression 参数是任意有效的表达式。 说明 通常,可
Chr 函数 返回与指定的 ANSI 字符代码相对应的字符。 Chr(charcode) charcode 参数是可以标识字符的数字。 说明 从 0 到 31 的数字表示标准的不可打印的
CDbl 函数 返回表达式,此表达式已被转换为 Double 子类型的 Variant。 CDbl(expression) expression 参数是任意有效的表达式。 说明 通常,您可
CDate 函数 返回表达式,此表达式已被转换为 Date 子类型的 Variant。 CDate(date) date 参数是任意有效的日期表达式。 说明 IsDate 函数用于判断 d
CCur 函数 返回表达式,此表达式已被转换为 Currency 子类型的 Variant。 CCur(expression) expression 参数是任意有效的表达式。 说明 通常,
CByte 函数 返回表达式,此表达式已被转换为 Byte 子类型的 Variant。 CByte(expression) expression 参数是任意有效的表达式。 说明 通常,可以
CBool 函数 返回表达式,此表达式已转换为 Boolean 子类型的 Variant。 CBool(expression) expression 是任意有效的表达式。 说明 如果 ex
Atn 函数 返回数值的反正切值。 Atn(number) number 参数可以是任意有效的数值表达式。 说明 Atn 函数计算直角三角形两个边的比值 (number) 并返回对应角的弧
Asc 函数 返回与字符串的第一个字母对应的 ANSI 字符代码。 Asc(string) string 参数是任意有效的字符串表达式。如果 string 参数未包含字符,则将发生运行时错误。
Array 函数 返回包含数组的 Variant。 Array(arglist) arglist 参数是赋给包含在 Variant 中的数组元素的值的列表(用逗号分隔)。如果没有指定此参数,则
Abs 函数 返回数字的绝对值。 Abs(number) number 参数可以是任意有效的数值表达式。如果 number 包含 Null,则返回 Null;如果是未初始化变量,则返回 0。
FormatPercent 函数 返回表达式,此表达式已被格式化为尾随有 % 符号的百分比(乘以 100 )。 FormatPercent(expression[,NumDigitsAfterD
FormatNumber 函数 返回表达式,此表达式已被格式化为数值。 FormatNumber( expression [,NumDigitsAfterDecimal [,Inc
我是一名优秀的程序员,十分优秀!