动态类型未清除时重新分配变量 - 这是否会出错?
来源:stackoverflow
时间:2024-02-08 20:54:03 441浏览 收藏
编程并不是一个机械性的工作,而是需要有思考,有创新的工作,语法是固定的,但解决问题的思路则是依靠人的思维,这就需要我们坚持学习和更新自己的知识。今天golang学习网就整理分享《动态类型未清除时重新分配变量 - 这是否会出错?》,文章讲解的知识点主要包括,如果你对Golang方面的知识点感兴趣,就不要错过golang学习网,在这可以对大家的知识积累有所帮助,助力开发能力的提升。
Go 中有一个众所周知的怪癖,即持有 nil 值的接口不等于 nil。这是因为在底层,接口是动态类型和值的组合,只有当两者都为 nil 时,它才为 nil。所以 (*MyStruct)(nil) != nil
但 (nil)(nil) == nil
。本博客对此进行了更好的解释。
我发现了一些与此行为相关的东西,这让我感到惊讶,在这里:https://goplay.tools/snippet/VF8oWt9XvO8。代码也复制如下。
看来,如果您重新分配分配了动态类型的变量,动态类型就会被记住并保留为新值。这对我来说似乎是出乎意料的,我认为重新分配变量应该覆盖所有过去的状态。
我检查了语言规范,但它有点含糊:https://go.dev/ref/spec#Assignability
Unlike regular variable declarations, a short variable declaration may redeclare variables provided they were originally declared earlier in the same block ... Redeclaration does not introduce a new variable; it just assigns a new value to the original.
不清楚这是否仅意味着值,或者值加上动态类型。
这种行为是语言中有意为之的,还是运行时重新使用内存进行变量重新分配而不清除所有状态的一些副作用?
代码:
package main import ( "fmt" ) type CustomError struct{} func (e *CustomError) Error() string { return "err" } // =================== func FuncThatReturnsCustomError() *CustomError { return nil } func FuncThatReturnsCustomErrorAsErrorInterface() error { // although the underlying returned value is nil, the return value from this func != nil // https://glucn.medium.com/golang-an-interface-holding-a-nil-value-is-not-nil-bb151f472cc7 return FuncThatReturnsCustomError() } // =================== func main() { // Assign a non-nil value to err (value is nil, dynamic type is not) err := FuncThatReturnsCustomErrorAsErrorInterface() fmt.Printf("err == nil: %v false because although the value is nil, the dynamic type is not nil (expected)\n", err == nil) // Call func where return type is a pointer to a struct and and returns nil // It is nil, as expected, this call is just to provide a comparison with the call after newlyDeclaredErr := FuncThatReturnsCustomError() fmt.Printf("newlyDeclaredErr == nil: %v true because func returned nil (expected)\n", newlyDeclaredErr == nil) // Exactly the same call as above, but reusing the original err variable instead of declaring a new one // Back to not nil, unexpected err = FuncThatReturnsCustomError() fmt.Printf("original err reassigned == nil: %v false presumably because err remembered its old dynamic type even after reassignment (unexpected)\n", err == nil) // Reassign err again, but explicitly to nil rather than by calling a function that returns nil. This time it's nil again err = nil fmt.Printf("original err after assignment to nil == nil: %v true, expected but not consistent with the case above\n", err == nil) }
正确答案
你“意想不到”的部分是这样的:
err = FuncThatReturnsCustomError()
您期望结果为 nil
。 err
是接口类型(error
)的变量,FuncThatReturnsCustomError()
的返回类型为 *CustomError
。这不是一个接口类型,而是一个具体类型(指向 CustomError
的指针)。由于它返回一个非接口值,因此当分配给接口类型的变量时,必须将其包装到接口值中。这是将创建非 nil
接口值的位置。这与“记住”或“保留”旧类型信息无关。
如果您使用具有如下接口结果类型的函数:
func returnNilErr() error { return nil }
并测试它:
err = returnNilErr() fmt.Printf("result of returnNilErr() == nil: %v\n", err == nil)
您会得到(在 Go Playground 上尝试一下):
result of returnNilErr() == nil: true
因为returnNilErr()
已经有接口结果类型(error
),所以它的返回值不需要打包成接口值,在赋值给err
变量时可以照原样使用。 p>
查看相关/可能的重复:隐藏 nil 值,理解为什么 Go 在这里失败
Go 常见问题解答:为什么我的 nil 错误值不等于 nil?一个>
以上就是《动态类型未清除时重新分配变量 - 这是否会出错?》的详细内容,更多关于的资料请关注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次学习