- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我已经实现了一个自定义错误类型,并且确实得到了关于 nil 值的奇怪行为。当我将自定义错误作为标准错误接口(interface)传递时,它永远不会被识别为 nil,即使自定义错误返回为 nil。
看看这个小测试程序:
package main
import (
"fmt"
"strconv"
)
type CustomError struct {
Code int
}
func (e *CustomError) Error() string {
return strconv.Itoa(e.Code)
}
func FailCustom(dofail bool) *CustomError {
if dofail {
return &CustomError{Code: 42}
} else {
return nil
}
}
func WrapFailCustom(dofail bool) error {
return FailCustom(dofail)
}
func main() {
err := WrapFailCustom(false)
if err == nil {
fmt.Println("err is nil")
} else {
fmt.Println("err is not nil")
}
}
同样在 Playground 上:https://play.golang.org/p/7bqeDw5B5fU
这实际上输出“err is not nil”。
我本以为 *CustomError 类型的 nil 值会隐式转换为 error 类型的 nil 值。谁能向我解释为什么不是这种情况以及如何正确传播自定义错误类型的 nil 值?
编辑:可以找到对此的解释 here正如伊恩·邓肯所指出的
为了进一步探讨这个问题,让我们考虑对 WrapFailCustom 进行以下修改:
func WrapFailCustom(dofail bool) error {
err := FailCustom(dofail)
if err == nil {
return nil
} else {
return err
}
}
这实际上返回“err is nil”:https://play.golang.org/p/mEKJFyk5zqf
我确实觉得依赖这个作为解决方案真的很糟糕,因为在使用会吐出我的自定义错误的函数时很容易忘记它。有没有更好的方法来制作自定义错误来防止这种“歧义”的发生?一直使用基本错误类型的明显解决方案,对于使用 WrapFailCustom 等函数的代码来说似乎真的很不方便,所以我想避免这种情况......
最佳答案
有关背景,请参阅 Hiding nil values, understanding why golang fails here ;和 Go FAQ: Why is my nil error value not equal to nil?
Go 要求您明确类型和转换(例如,您不能将类型 int32
的值添加到类型 int
的值),但是接口(interface)转换自动创建接口(interface)值是这个规则的异常(exception)。每当需要接口(interface)类型的值时,您可以使用其类型实现(满足)接口(interface)类型的任何值,接口(interface)值将自动为您创建。
你的功能:
func WrapFailCustom(dofail bool) error {
return FailCustom(dofail)
}
WrapFailCustom()
有一个返回类型error
,你尝试返回 FailCustom()
函数的结果,其返回类型是 *CustomError
与 error
不同!这应该引起警惕!
什么/如何返回? error
类型的接口(interface)值将被自动创建! *CustomError
实现了 error
,所以一切都很好,但是正如您所经历的,如果指针是 nil
,这种隐式值包装不会导致一个接口(interface)值是 nil
,而是一个非 nil
接口(interface)值包装了值 nil
和类型 *CustomError
.
解决方案?
FailCustom()
有一个不同于error
的返回类型真的合理/它有任何值(value)吗?如果没有,最简单的方法是在“问题”的来源处处理它:
func FailCustom(dofail bool) error {
if dofail {
return &CustomError{Code: 42}
}
return nil
}
然后您所有的问题都会消失。如果您按照“Go 方式”使用 error
类型返回错误,这就足够了并且令人满意。您甚至不再需要 WrapFailCustom()
函数。
WrapFailCustom()
如果您确实需要 FailCustom()
返回自定义 *CustomError
类型,那么您需要在 WrapFailCustom()< 中“手动”考虑它
。我会这样写:
func WrapFailCustom(dofail bool) error {
if customErr := FailCustom(dofail); customErr != nil {
return customErr
}
return nil
}
(请注意,我故意使用不同的 customErr
名称而不是 err
,表明它不是 error
类型,应该注意采取如何将其转换为error
。)
错误
类型如果您想/需要使用您的自定义错误类型,另一种好方法是创建一个接口(interface)类型来描述它包含的“额外”功能:
type CustomErr interface {
Error // Embed error interface
Code() int
}
然后我们还需要实现这个Code()
方法:
func (e *CustomError) Code() int { return e.Code }
这个有什么用?
根本原因将通过返回此接口(interface) 类型(并且不是 指针)的值来处理:
func FailCustom(dofail bool) CustomErr {
if dofail {
return &CustomError{Code: 42}
}
return nil
}
隐式接口(interface)值将在 FailCustom()
中创建。
此外,WrapFailCustom()
变得不必要/无用。 FailCustom()
返回一个既是错误
又是代码
的值,您可以使用它的代码()<从中获取
方法。返回值是一个接口(interface)值,是一个代码
/error
,你可以在需要error
值的地方使用它。具体类型 CustomError
甚至可以不导出(隐藏)。
与此方法相关,查看 Dave Cheney: Don’t just check errors, handle them gracefully ,尤其是标题为:Assert errors for behavior, not type 的部分。
关于pointers - 如何在不导致 nil 值问题的情况下返回自定义错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49768691/
我是 Java 新手,这是我的代码, if( a.name == b.name && a.displayname == b.displayname && a.linknam
在下面的场景中,我有一个 bool 值。根据结果,我调用完全相同的函数,唯一的区别是参数的数量。 var myBoolean = ... if (myBoolean) { retrieve
我是一名研究 C++ 的 C 开发人员: 我是否正确理解如果我抛出异常然后堆栈将展开直到找到第一个异常处理程序?是否可以在不展开的情况下在任何 throw 上打开调试器(即不离开声明它的范围或任何更高
在修复庞大代码库中的错误时,我观察到一个奇怪的情况,其中引用的动态类型从原始 Derived 类型更改为 Base 类型!我提供了最少的代码来解释问题: struct Base { // some
我正在尝试用 C# 扩展给定的代码,但由于缺乏编程经验,我有点陷入困境。 使用 Visual Studio 社区,我尝试通过控制台读出 CPU 核心温度。该代码使用开关/外壳来查找传感器的特定名称(即
这可能是一个哲学问题。 假设您正在向页面发出 AJAX 请求(这是使用 Prototype): new Ajax.Request('target.asp', { method:"post", pa
我有以下 HTML 代码,我无法在所有浏览器中正常工作: 我试图在移动到
我对 Swift 很陌生。我如何从 addPin 函数中检索注释并能够在我的 addLocation 操作 (buttonPressed) 中使用它。我正在尝试使用压力触摸在 map 上添加图钉,在两
我设置了一个详细 View ,我是否有几个 Nib 文件根据在 Root View Controller 的表中选择的项目来加载。 我发现,对于 Nibs 的类,永远不会调用 viewDidUnloa
我需要动态访问 json 文件并使用以下代码。在本例中,“bpicsel”和“temp”是变量。最终结果类似于“data[0].extit1” var title="data["+bpicsel+"]
我需要使用第三方 WCF 服务。我已经在我的证书存储中配置了所需的证书,但是在调用 WCF 服务时出现以下异常。 向 https://XXXX.com/AHSharedServices/Custome
在几个 SO 答案(1、2)中,建议如果存在冲突则不应触发 INSERT 触发器,ON CONFLICT DO NOTHING 在触发语句中。也许我理解错了,但在我的实验中似乎并非如此。 这是我的 S
如果进行修改,则会给出org.hibernate.NonUniqueObjectException。在我的 BidderBO 类(class)中 @Override @Transactional(pr
我使用 indexOf() 方法来精细地查找数组中的对象。 直到此刻我查了一些资料,发现代码应该无法正常工作。 我在reducer中尝试了上面的代码,它成功了 let tmp = state.find
假设我有以下表格: CREATE TABLE Game ( GameID INT UNSIGNED NOT NULL, GameType TINYINT UNSIGNED NOT NU
代码: Alamofire.request(URL(string: imageUrl)!).downloadProgress(closure: { (progress) in
我是一名优秀的程序员,十分优秀!