这个代码竟然报错了?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)  }  令我惊讶的是

  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] 中提到过)

目录
相关文章
|
12天前
|
程序员 测试技术 Go
用 Go 编写简洁代码的最佳实践
用 Go 编写简洁代码的最佳实践
|
12天前
|
缓存 测试技术 Go
使用Singleflight优化Go代码
使用Singleflight优化Go代码
|
11天前
|
JSON 数据库连接 Go
10个令人惊叹的Go语言技巧,让你的代码更加优雅
10个令人惊叹的Go语言技巧,让你的代码更加优雅
|
16天前
|
Go
Go 语言中的“继承”:使用结构体实现代码重用
Go 语言中的“继承”:使用结构体实现代码重用
22 0
|
22天前
|
Go PHP 云计算
如何在PHP代码里面调用Go程序
PHP和Go都是广泛使用的编程语言,但各自具有其独特的优势和特性。
|
1月前
|
分布式计算 大数据 Go
MaxCompute操作报错合集之使用go sdk调用GetTunnelEndpoint出现报错:InvalidAction.NotFoundSpecified api is not found,该如何解决
MaxCompute是阿里云提供的大规模离线数据处理服务,用于大数据分析、挖掘和报表生成等场景。在使用MaxCompute进行数据处理时,可能会遇到各种操作报错。以下是一些常见的MaxCompute操作报错及其可能的原因与解决措施的合集。
|
1月前
|
运维 Devops API
阿里云云效操作报错合集之直接用API调用可以使用,但是本地用Go代码调用失败,是什么导致的?
本合集将整理呈现用户在使用过程中遇到的报错及其对应的解决办法,包括但不限于账户权限设置错误、项目配置不正确、代码提交冲突、构建任务执行失败、测试环境异常、需求流转阻塞等问题。阿里云云效是一站式企业级研发协同和DevOps平台,为企业提供从需求规划、开发、测试、发布到运维、运营的全流程端到端服务和工具支撑,致力于提升企业的研发效能和创新能力。
|
1天前
|
安全 Java Go
探索Go语言在高并发环境中的优势
在当今的技术环境中,高并发处理能力成为评估编程语言性能的关键因素之一。Go语言(Golang),作为Google开发的一种编程语言,以其独特的并发处理模型和高效的性能赢得了广泛关注。本文将深入探讨Go语言在高并发环境中的优势,尤其是其goroutine和channel机制如何简化并发编程,提升系统的响应速度和稳定性。通过具体的案例分析和性能对比,本文揭示了Go语言在实际应用中的高效性,并为开发者在选择合适技术栈时提供参考。