这个代码竟然报错了?Go 的不可寻址值了解下

Go语言精选

共 2779字,需浏览 6分钟

 ·

2020-07-27 19:49

点击上方蓝色“Go语言中文网”关注我们,领全套Go资料,每天学习 Go 语言

Dave Cheney 在 Twitter 上发布过一个 Go 的小测验,和往常一样,我从中学到了一些有趣的东西。让我们从他的推文开始:

#golang 小测验:该程序打印什么?

package main
import (
    "crypto/sha1"
    "fmt"
)

func main() {
    input := []byte("Hello, playground")
    hash := sha1.Sum(input)[:5]
    fmt.Println(hash)
}

令我惊讶的是,答案是:

./test.go:10:28: invalid operation sha1.Sum(input)[:5] (slice of unaddressable value)

我们收到此错误有三个原因。首先,`sha1.Sum()`[1] 的返回值不寻常。大多数方法返回切片,而此代码对切片不会报错。但是 sha1.Sum() 返回的值很奇怪,它是一个固定大小的数组(具体来说是 [20]byte ),由于 Go 是返回数值的,这意味着它确实向 main() 返回了 20 字节的数组,而不是指向它的指针。

这就涉及到了不可寻址值的概念,与可寻址值相反。详细的介绍在 Go 编程语言规范的 地址运算符[2] 中。简单来说,大多数匿名值都不可寻址( 复合字面值[3] 是一个大大的例外)。在上面的代码中,sha1.Sum() 的返回值是匿名的,因为我们立即对其进行了切片操作。如果我们将它存在变量中,并因此使其变为非匿名,则该代码不会报错:

tmp := sha1.Sum(input)
hash := tmp[:5]

最后一个问题是为什么切片操作是错误的。这是因为对数组进行切片操作要求该数组是可寻址的(在 Go 编程语言规范的 Slice 表达式[4] 的末尾介绍)。sha1.Sum() 返回的匿名数组是不可寻址的,因此对其进行切片会被编译器拒绝。

(将返回值存储到我们的 tmp 变量中使其变成了可寻址。sha1.Sum() 的返回值在复制到 tmp 后就消失了。)

虽然我不能完全理解为什么 Go 的设计师限制了哪些值是可寻址的,但是我可以想到几条原因。例如,如果在这里允许切片操作,那么 Go 会默默地实现堆存储以容纳 sha1.Sum() 的返回值(然后将该值复制到另一个值),该返回值将一直存在直到那个切片被回收。

(如 x86-64 上的 Go 低级调用惯例[5] 中所述,由于 Go 返回了栈中的所有值,因此需要将数据进行拷贝。对于 sha1.Sum() 的 20 字节的返回值来说,这并不是什么大事。我很确定人们经常使用更大的结构体作为返回值。)

PS:Go 语言规范中的许多内容要求或仅对可寻址的值适用。例如,大多数 赋值[6] 操作需要可寻址性。

补充:方法调用和可寻址性

假设有一个类型 T,并且在 *T 上定义了一些方法,例如 *T.Op()。就像 Go 允许在不取消引用指针的情况下进行字段引用一样,你可以在非指针值上调用指针方法:

var x T
x.Op()

这是 (&x).Op() 的简便写法(在 Go 编程语言规范文中靠后的 调用[7] 部分进行了介绍)。但是,由于此简便写法需要获取地址,因此需要可寻址性。因此,以下操作会报错:

// afunc() 返回一个 T
afunc().Op()

// 但是这个可以运行:
var x T = afunc()
x.Op()

之前我已经看到人们在讨论 Go 在方法调用上的怪癖,但是当时我还不完全了解发生了什么,以及由于什么原因使方法调用无法正常工作。

(请注意,这种简写转换与 *T 具有所有 T 方法是根本不同的,这在我 之前的一篇文章[8] 中提到过)


via: https://utcc.utoronto.ca/~cks/space/blog/programming/GoUnaddressableSlice

作者:Chris wSiebenmann[9]译者:zhiyu-tracy-yang[10]校对:polaris1119[11]

本文由 GCTT[12] 原创编译,Go 中文网[13] 荣誉推出

参考资料

[1]

sha1.Sum(): https://golang.org/pkg/crypto/sha1/#Sum

[2]

地址运算符: https://golang.org/ref/spec#Address_operators

[3]

复合字面值: https://golang.org/ref/spec#Composite_literals

[4]

Slice 表达式: https://golang.org/ref/spec#Slice_expressions

[5]

x86-64 上的 Go 低级调用惯例: https://science.raphael.poss.name/go-calling-convention-x86-64.html#arguments-and-return-value

[6]

赋值: https://golang.org/ref/spec#Assignments

[7]

调用: https://golang.org/ref/spec#Calls

[8]

之前的一篇文章: https://utcc.utoronto.ca/~cks/space/blog/programming/GoInterfacesAutogenFuncs

[9]

Chris wSiebenmann: https://utcc.utoronto.ca/~cks/space/People/ChrisSiebenmann

[10]

zhiyu-tracy-yang: https://github.com/zhiyu-tracy-yang

[11]

polaris1119: https://github.com/polaris1119

[12]

GCTT: https://github.com/studygolang/GCTT

[13]

Go 中文网: https://studygolang.com/



推荐阅读


学习交流 Go 语言,扫码回复「进群」即可


站长 polarisxu

自己的原创文章

不限于 Go 技术

职场和创业经验


Go语言中文网

每天为你

分享 Go 知识

Go爱好者值得关注


浏览 36
点赞
评论
收藏
分享

手机扫一扫分享

举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

举报