区分零切片是什么意思,即。未初始化的切片和一个空切片,即。初始化但空片?
我理解其中的差异,但我想知道两种情况之间存在细微差别的动机是什么?对于所有意图和目的,使用nil切片和空切片时表现相同。
似乎Go开发人员只有一个案例,例如只允许空切片,它会简化心理模型并消除微妙错误的来源。
是否有理由创建这两个用例?
答案 0 :(得分:3)
nil
切片值不需要分配。这可能会在您想要在切片中构建某些内容的情况下产生影响,但通常不会附加数据,因此切片可能会保留nil
,因此不需要完全分配。
空切片可能需要分配,即使其容量为零。
空片也意味着它的长度为0,但其容量可能不是;所以"并非如此;对于所有意图和目的,nil
切片和空切片在使用它们时表现相同。" 。您可以分配0长度和大容量的切片,优化进一步的附加以避免分配(和复制):
s := make([]int, 0)
fmt.Println(s, len(s), cap(s))
s = append(s, 1)
fmt.Println(s, len(s), cap(s))
s = make([]int, 0, 10)
fmt.Println(s, len(s), cap(s))
s = append(s, 1)
fmt.Println(s, len(s), cap(s))
上述输出(在Go Playground上尝试):
[] 0 0
[1] 1 2
[] 0 10
[1] 1 10
我们看到了什么?在第一个例子中,我们创建了一个0长度和0容量的空切片。如果我们向它追加一个元素,它的长度将变为1(显然),并且它的容量增加到2.这是因为在引擎盖append()
下分配了一个大小为2的新数组(考虑未来增长),复制了现有元素(在本例中为none),并分配了新元素。
在第二种情况下,我们从空切片开始,但容量为10.这意味着我们可以向其追加10个元素,而不会导致新的分配和复制现有元素。当切片很大时,这可能是一个很大的优势,这需要多次完成。