Go 语言中的 nil 切片 vs 非 nil 切片 vs 空切片
来源:Golang技术栈
时间:2023-03-05 11:10:23 162浏览 收藏
大家好,我们又见面了啊~本文《Go 语言中的 nil 切片 vs 非 nil 切片 vs 空切片》的内容中将会涉及到golang等等。如果你正在学习Golang相关知识,欢迎关注我,以后会给大家带来更多Golang相关文章,希望我们能一起进步!下面就开始本文的正式内容~
问题内容
我是 Go 编程的新手。我在 Go 编程书籍中读到 slice 由三部分组成:指向数组的指针、长度和容量。
我在 nil 切片(切片没有要指向的底层数组,len = 0,cap = 0)、只有 len = 0、cap = 0 的非 nil 切片和空切片之间感到困惑。
谁能告诉 nil 和 empty slices 是否相同?如果它们都不同,那么请告诉它们两者之间的区别是什么?
如何测试切片是否为空?
另外,指针在长度和容量为零的非零切片中保存什么值?
正确答案
可观察的行为
nil
和空切片(容量为 0)不一样,但它们的可观察行为是相同的。我的意思是:
- 您可以将它们传递给内置函数
len()
和cap()
函数 - 你可以
for range
超越它们(将是 0 次迭代) - 您可以对它们进行切片(通过不违反规范中概述的限制:切片表达式;因此结果也将是一个空切片)
- 由于它们的长度为 0,因此您不能更改它们的内容(附加一个值会创建一个新的切片值)
看这个简单的例子(一个nil
切片和 2 个非nil
空切片):
var s1 []int // nil slice s2 := []int{} // non-nil, empty slice s3 := make([]int, 0) // non-nil, empty slice fmt.Println("s1", len(s1), cap(s1), s1 == nil, s1[:], s1[:] == nil) fmt.Println("s2", len(s2), cap(s2), s2 == nil, s2[:], s2[:] == nil) fmt.Println("s3", len(s3), cap(s3), s3 == nil, s3[:], s3[:] == nil) for range s1 {} for range s2 {} for range s3 {}
输出(在Go Playground上试试):
s1 0 0 true [] true s2 0 0 false [] false s3 0 0 false [] false
(请注意,切片nil
会产生nil
切片,切片非nil
切片会产生非nil
切片。)
您只能通过将切片值与预先声明的标识符进行比较来区分nil
它们,它们在其他方面的行为相同。
要判断一个切片是否为空,只需将其长度与0
:进行比较len(s) == 0
。不管是nil
切片还是非nil
切片,它是否具有正容量也没关系;如果它没有元素,它是空的。
s := make([]int, 0, 100) fmt.Println("Empty:", len(s) == 0, ", but capacity:", cap(s))
打印(在Go Playground上尝试):
Empty: true , but capacity: 100
在引擎盖下
切片值由定义在
中的结构体表示reflect.SliceHeader
:
type SliceHeader struct { Data uintptr Len int Cap int }
在nil
切片的情况下,此结构将具有其零值,即它的所有字段都将是它们的零值,即:0
。
拥有一个nil
容量和长度都等于 的非切片0
,Len
并且Cap
字段肯定是0
,但Data
指针可能不是。它 不会
,这就是它与nil
切片的区别。它将指向一个大小为零的底层数组。
请注意,Go 规范允许大小为 0 的不同类型的值具有相同的内存地址。规格:系统注意事项:尺寸和对齐保证:
如果结构或数组类型不包含大小大于零的字段(或元素),则其大小为零。 两个不同的零大小变量在内存中可能具有相同的地址。
让我们检查一下。为此,我们调用unsafe
包的帮助,并“获取”reflect.SliceHeader
切片值的结构“视图”:
var s1 []int s2 := []int{} s3 := make([]int, 0) fmt.Printf("s1 (addr: %p): %+8v\n", &s1, *(*reflect.SliceHeader)(unsafe.Pointer(&s1))) fmt.Printf("s2 (addr: %p): %+8v\n", &s2, *(*reflect.SliceHeader)(unsafe.Pointer(&s2))) fmt.Printf("s3 (addr: %p): %+8v\n", &s3, *(*reflect.SliceHeader)(unsafe.Pointer(&s3)))
输出(在Go Playground上试试):
s1 (addr: 0x1040a130): {Data: 0 Len: 0 Cap: 0} s2 (addr: 0x1040a140): {Data: 1535812 Len: 0 Cap: 0} s3 (addr: 0x1040a150): {Data: 1535812 Len: 0 Cap: 0}
我们看到了什么?
- 所有切片(切片头)具有不同的内存地址
nil
切片有数据0
指针s2
和s3
切片确实具有相同的数据指针,共享/指向相同的 0 大小的内存值
今天关于《Go 语言中的 nil 切片 vs 非 nil 切片 vs 空切片》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!
-
439 收藏
-
262 收藏
-
193 收藏
-
188 收藏
-
500 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习