- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我想知道在 go 中处理多级抽象错误的最佳方法是什么。每次如果我必须为程序添加一个新的层次抽象,我就被迫将错误代码从较低的层次转移到较高的层次。因此,日志文件中存在重复通信,否则我必须记住删除低级别的通信表单并将其转移到更高级别。下面简单举例。我跳过了创建每个对象到更简短的代码,但我想你理解我的问题
type ObjectOne struct{
someValue int
}
func (o* ObjectOne)CheckValue()error{
if o.someValue == 0 {
SomeLogger.Printf("Value is 0 error program") // communicate form first level abstraction to logger
return errors.New("Internal value in object is 0")
}
return nil
}
type ObjectTwoHigherLevel struct{
objectOne ObjectOne
}
func (oT* ObjectTwoHigherLevel)CheckObjectOneIsReady() error{
if err := oT.objectOne.CheckValue() ; err != nil{
SomeLogger.Printf("Value in objectOne is not correct for objectTwo %s" , err) // second communicate
return err
}
return nil
}
type ObjectThreeHiggerLevel struct{
oT ObjectTwoHigherLevel
}
func (oTh* ObjectThreeHiggerLevel)CheckObjectTwoIsReady()error{
if err := oTh.oT.CheckObjectOneIsReady() ; err != nil{
SomeLogger.Printf("Value in objectTwo is not correct for objectThree %s" , err)
return err
}
return nil
}
在日志文件中我得到了重复的帖子
Value is 0 error program
Value in objectOne is not correct for objectTwo Internal value in object is 0
Value in objectTwo is not correct for objectThree Internal value in object is 0
反过来,如果我只将一些 err
传输到更高级别而没有额外的日志,我会丢失每个级别发生的信息。
这个怎么解决?私有(private)副本如何通信?或者我的方式是唯一的好办法?
如果我创建了一些对象来在一些抽象级别上在数据库中搜索某些内容,那么问题就更令人沮丧了,然后我在 logFile 中也从这个相同的任务中得到几行。
最佳答案
编辑: 这个答案早于 Go 1.13,它提供了与所呈现的技术类似的东西。请查看The Go Blog: Working with Errors in Go 1.13 .
您应该处理错误,或者不处理错误而是将其委托(delegate)给更高级别(调用者)。处理错误并将其返回是不好的做法,就好像调用者也做同样的事情一样,错误可能会被处理多次。
处理错误意味着检查它并据此做出决定,这可能是您简单地记录它,但这也算作“处理”它。
如果您选择不处理而是将其委托(delegate)给更高级别,那可能完全没问题,但不要只返回您得到的错误值,因为它可能对没有上下文的调用者毫无意义。
注释错误是一种非常好的和推荐的委派方式。这意味着您创建并返回了一个新 错误值,但旧值也包含在返回值中。包装器为包装错误提供上下文。
有一个用于注释错误的公共(public)库:github.com/pkg/errors
;及其 godoc:errors
它基本上有 2 个功能:1 个用于 wrapping现有错误:
func Wrap(cause error, message string) error
还有一个用于 extracting包装错误:
func Cause(err error) error
使用这些,这就是您的错误处理方式:
func (o *ObjectOne) CheckValue() error {
if o.someValue == 0 {
return errors.New("Object1 illegal state: value is 0")
}
return nil
}
第二层:
func (oT *ObjectTwoHigherLevel) CheckObjectOneIsReady() error {
if err := oT.objectOne.CheckValue(); err != nil {
return errors.Wrap(err, "Object2 illegal state: Object1 is invalid")
}
return nil
}
第三级:只调用第二级检查:
func (oTh *ObjectThreeHiggerLevel) CheckObjectTwoIsReady() error {
if err := oTh.ObjectTwoHigherLevel.CheckObjectOneIsReady(); err != nil {
return errors.Wrap(err, "Object3 illegal state: Object2 is invalid")
}
return nil
}
请注意,由于 CheckXX()
方法不处理错误,因此它们不会记录任何内容。他们委托(delegate)带注释的错误。
如果有人使用 ObjectThreeHiggerLevel
决定处理错误:
o3 := &ObjectThreeHiggerLevel{}
if err := o3.CheckObjectTwoIsReady(); err != nil {
fmt.Println(err)
}
将呈现以下漂亮的输出:
Object3 illegal state: Object2 is invalid: Object2 illegal state: Object1 is invalid: Object1 illegal state: value is 0
没有多个日志的污染,并且所有的细节和上下文都被保留,因为我们使用了 errors.Wrap()
产生一个错误值,该错误值被格式化为一个 string
以递归方式保留包装的错误:错误堆栈。
您可以在博文中阅读有关此技术的更多信息:
Dave Cheney: Don’t just check errors, handle them gracefully
如果您喜欢更简单的事情和/或您不想与外部库打交道,并且您可以接受无法提取原始错误(确切的错误值,而不是error string,你可以),那么你可以简单地使用上下文扩展错误并返回这个新的扩展错误。
使用 fmt.Errorf()
最容易扩展错误它允许您创建一个“漂亮”的格式化错误消息,并返回一个类型为 error
的值,因此您可以直接返回它。
使用 fmt.Errorf()
,错误处理可能如下所示:
func (o *ObjectOne) CheckValue() error {
if o.someValue == 0 {
return fmt.Errorf("Object1 illegal state: value is %d", o.someValue)
}
return nil
}
第二层:
func (oT *ObjectTwoHigherLevel) CheckObjectOneIsReady() error {
if err := oT.objectOne.CheckValue(); err != nil {
return fmt.Errorf("Object2 illegal state: %v", err)
}
return nil
}
第三级:只调用第二级检查:
func (oTh *ObjectThreeHiggerLevel) CheckObjectTwoIsReady() error {
if err := oTh.ObjectTwoHigherLevel.CheckObjectOneIsReady(); err != nil {
return fmt.Errorf("Object3 illegal state: %v", err)
}
return nil
}
如果它决定“处理”它,则以下错误消息将显示在 ObjectThreeHiggerLevel
上:
o3 := &ObjectThreeHiggerLevel{}
if err := o3.CheckObjectTwoIsReady(); err != nil {
fmt.Println(err)
}
将呈现以下漂亮的输出:
Object3 illegal state: Object2 illegal state: Object1 illegal state: value is 0
请务必阅读博文:Error handling and Go
关于go - 处理来自多个抽象级别的错误的最佳实践,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37346694/
Android 项目中最低(最低 sdk)和最高(目标 sdk)级别是否有任何影响。这些东西是否会影响项目的可靠性和效率。 最佳答案 没有影响,如果您以 SDK 级别 8 为目标,那么您的应用将以 9
我将现有的 android 项目升级到 API 级别 31。我使用 Java 作为语言。我改变了 build.gradle compileSdkVersion 31 defaultConfig {
我正在使用 ionic 3 创建一个 android 应用程序,当我尝试上传到 playstore 时,我收到一个错误,提示我的应用程序以 api 25 为目标,当我检查我的 project.prop
我刚刚尝试将应用程序的目标和编译 API 级别更新为 29 (Android 10),并注意到我无法再编译,因为 LocationManager.addNmeaListener 只接受 OnNmeaM
我的代码没有在 Kitkat 上显示工具栏。 这是我的两个 Android 版本的屏幕截图。 Kitkat 版本: Lollipop 版: 这背后的原因可能是什么。 list 文件
我正在构建面向 API 级别 8 的 AccessabilityService,但我想使用 API 级别 18 中引入的功能 (getViewIdResourceName())。这应该可以通过使用 A
当我想在我的电脑上创建一个 android 虚拟机时,有两个选项可以选择目标设备。它们都用于相同的 API 级别。那么我应该选择哪一个呢?它们之间有什么区别? 最佳答案 一个是基本的 Android,
当我选择 tagret 作为 Android 4.2.2(API 级别 17)时,模拟器需要很长时间来加载和启动。 所以我研究它并通过使用 找到了解决方案Intel Atom(x86) 而不是 ARM
我有一个使用 Android Studio 创建的 Android 项目。我在项目中添加了一些第三方依赖项,但是当我尝试在 Android Studio 中编译时,我遇到了以下错误: Error:Ex
如上所述,如何使用 API 8 获取移动设备网络接口(interface)地址? 最佳答案 NetworkInterface.getInetAddresses() 在 API8 中可用。 关于andr
我想显示 Snackbar并使用图像而不是文本进行操作。 我使用以下代码: val imageSpan = ImageSpan(this, R.drawable.star) val b
我有一个用 python 编写的简单命令行程序。程序使用按以下方式配置的日志记录模块将日志记录到屏幕: logging.basicConfig(level=logging.INFO, format='
使用下面的代码,实现游戏状态以控制关卡的最简单和最简单的方法是什么?如果我想从标题画面开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人能解释处理这个问题的最简单方法,那就太好了! impor
我想创建一个可以找到嵌套树结构深度的属性。下面的静态通过递归找出深度/级别。但是是否可以将此函数作为同一个类中的属性而不是静态方法? public static int GetDepth(MenuGr
var myArray = [{ title: "Title 1", children: [{ title: "Title 1.1", children: [{
通过下面的代码,实现游戏状态来控制关卡的最简单、最容易的方法是什么?如果我想从标题屏幕开始,然后加载一个关卡,并在完成后进入下一个关卡?如果有人可以解释处理这个问题的最简单方法,那就太好了! impo
我有一个树结构,其中每个节点基本上可以有无限个子节点,它正在为博客的评论建模。 根据特定评论的 ID,我试图找出该评论在树中的深度/级别。 我正在关注 this guide that explains
考虑任何给定的唯一整数的数组,例如[1,3,2,4,6,5] 如何确定“排序度”的级别,范围从 0.0 到 1.0 ? 最佳答案 一种方法是评估必须移动以使其排序的项目数量,然后将其除以项目总数。 作
我如何定义一个模板类,它提供一个整数常量,表示作为输入模板参数提供的(指针)类型的“深度”?例如,如果类名为 Depth,则以下内容为真: Depth::value == 3 Depth::value
我的场景是:文件接收器应该包含所有内容。另一个接收器应包含信息消息,但需要注意的是 Microsoft.* 消息很烦人,因此这些消息应仅限于警告。两个sink怎么单独配置?我尝试的第一件事是: str
我是一名优秀的程序员,十分优秀!