Go 语言中的切片分类:nil 切片、非 nil 切片、空切片
来源:stackoverflow
时间:2024-02-08 16:27:43 121浏览 收藏
大家好,今天本人给大家带来文章《Go 语言中的切片分类:nil 切片、非 nil 切片、空切片》,文中内容主要涉及到,如果你对Golang方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!
问题内容我是 Go 编程的新手。我在Go编程书中读到,切片由三部分组成:指向数组的指针、长度和容量。
我对以下内容感到困惑:
- nil 切片(切片没有指向的底层数组,len = 0,cap=0)
- 仅 len = 0、cap = 0 的非零切片
- 空切片。
谁能告诉我 nil 和空切片是否是同一件事? 如果两者不同,那么请告诉我这两者有什么区别?如何测试切片是否为空?另外,指针在长度和容量为零的非零切片中保存什么值?
正确答案
可观察的行为
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
进行比较来区分,它们在其他方面的行为都是相同的。 但请注意,许多软件包确实会将切片与 nil
进行比较,并且可能会基于此进行不同的操作(例如 encoding/ json
和 fmt
包)。
唯一的区别是将切片转换为数组指针(已添加到Go 1.17 中的语言)。将非 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 大小的内存值
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习