当前位置:首页 > 科技  > 软件

利用 Golang 中的 Recover 处理错误

来源: 责编: 时间:2023-11-30 09:28:07 331观看
导读Golang 中的 recover 是一个鲜为人知但非常有趣和强大的功能。让我们看看它是如何工作的,以及在 Outreach.io 中如何利用它来处理 Kubernetes 中的错误。Panic/Defer/Recover 基本上是 Golang 中对于其他编程语言中 th

Golang 中的 recover 是一个鲜为人知但非常有趣和强大的功能。让我们看看它是如何工作的,以及在 Outreach.io 中如何利用它来处理 Kubernetes 中的错误。SqJ28资讯网——每日最新资讯28at.com

Panic/Defer/Recover 基本上是 Golang 中对于其他编程语言中 throw/finally/catch 概念的替代品。它们有一些共同之处,但在一些重要细节上有所不同。SqJ28资讯网——每日最新资讯28at.com

Defer

要充分理解 recover,我们首先需要谈论 defer 语句。defer 关键字前置于函数调用之前,使得该调用在当前函数返回之前执行。当我们在一个函数中使用多个 defer 语句时,它们按照后进先出的顺序执行,这使得创建清理逻辑变得非常容易,如下例所示:SqJ28资讯网——每日最新资讯28at.com

package mainimport (    "context"    "database/sql"    "fmt")func readRecords(ctx context.Context) error {    db, err := sql.Open("sqlite3", "file:test.db?cache=shared&mode=memory")    if err != nil {        return err    }    defer db.Close() // 这个函数调用将在 readRecords 函数返回时第三个执行    conn, err := db.Conn(ctx)    if err != nil {        return err    }    defer conn.Close() // 这个函数调用将在第二个执行    rows, err := conn.QueryContext(ctx, "SELECT id FROM users")    if err != nil {        return err    }    defer rows.Close() // 这个函数调用将在第一个执行    for rows.Next() {        var id int64        if err := rows.Scan(&id); err != nil {            return err        }        fmt.Println("ID:", id)    }    return nil}func main() {    readRecords(context.Background())}

Panic

我们需要谈论的第二个主题是 panic,它是一个导致当前 goroutine 进入 panic 模式的函数。当前函数中的正常执行流程被停止,仅执行 defer 语句,然后对调用者函数执行相同的操作,因此一直冒泡到堆栈的顶部(main 函数),然后使程序崩溃。panic 可以直接调用(传递一个值作为参数),也可以由运行时错误引起。例如,由于空指针解引用:SqJ28资讯网——每日最新资讯28at.com

package mainimport "fmt"func main() {    var x *string    fmt.Println(*x)}// panic: runtime error: invalid memory address or nil pointer dereference

Recover

recover 是一个内建函数,它使我们有可能在发生 panic 时重新获得控制。它仅在被调用的延迟函数中产生效果。在延迟函数之外调用时,它总是返回 nil。如果我们处于 panic 模式,调用 recover 会返回传递给 panic 函数的值。基本示例:SqJ28资讯网——每日最新资讯28at.com

package mainimport "fmt"func main() {    defer func() {        if r := recover(); r != nil {            fmt.Printf("Recovered: %v//n", r)        }    }()    panic("spam, egg, sausage, and spam")}// Recovered: spam, egg, sausage, and spam

我们可以以同样的方式从运行时错误中恢复:SqJ28资讯网——每日最新资讯28at.com

package mainimport "fmt"func main() {    defer func() {        if r := recover(); r != nil {            fmt.Printf("Recovered: %v//n", r)        }    }()    var x *string    fmt.Println(*x)}// Recovered: runtime error: invalid memory address or nil pointer dereference

在这种情况下,recover 返回的值的类型是错误(更准确地说是 runtime.errorString)。SqJ28资讯网——每日最新资讯28at.com

有一个限制:我们不能直接从 recover 块中返回值,因为在 recover 块中的 return 语句仅从延迟函数中返回,而不是从周围的函数中返回:SqJ28资讯网——每日最新资讯28at.com

package mainimport "fmt"func foo() int {    defer func() {        if r := recover(); r != nil {            fmt.Printf("Recovered: %v//n", r)            return 1 // "too many return values" 因为我们仅从匿名函数返回        }    }()    panic("spam, egg, sausage, and spam")}func main() {    x := foo()    fmt.Println(x)}

如果我们想要更改函数返回的值,我们需要使用命名返回值:SqJ28资讯网——每日最新资讯28at.com

package mainimport "fmt"func foo() (ret int) {    defer func() {        if r := recover(); r != nil {            fmt.Printf("Recovered: %v//n", r)            ret = 1        }    }()    panic("spam, egg, sausage, and spam")}func main() {    x := foo()    fmt.Println("value:", x)}// Recovered: spam, egg, sausage, and spam// value: 1

一个更实际的例子,将 panic 转换为普通错误的转换可能如下所示:SqJ28资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "github.com/google/uuid")// processInput 尝试将输入字符串转换为 uuid.UUID// 它将 panic 转换为错误func processInput(input string) (u uuid.UUID, err error) {    defer func() {        if r := recover(); r != nil {            err = fmt.Errorf("panic: %v", r)        }    }()    // 一些可能引发 panic 的逻辑(也可以是第三方逻辑),例如:    u = uuid.MustParse(input)    return u, nil}func main() {    u, err := processInput("xxx")    if err != nil {        fmt.Println(err)    }    fmt.Println(u)}// panic: uuid: Parse(xxx): invalid UUID length: 3// 00000000-0000-0000-0000-000000000000

现在让我们尝试一些稍微SqJ28资讯网——每日最新资讯28at.com

复杂的东西。假设我们在 Kubernetes 中运行,并且我们想要编写一个通用的 recover 函数,处理所有未捕获的 panic 和运行时错误,并收集它们的堆栈跟踪,以便我们可以以结构化的方式记录它们(例如,以 JSON 格式)。SqJ28资讯网——每日最新资讯28at.com

package mainimport (    "fmt"    "log"    "os"    "github.com/pkg/errors")func foo() string {    var s *string    return *s}func handlePanic(r interface{}) error {    var errWithStack error    if err, ok := r.(error); ok {        errWithStack = errors.WithStack(err)    } else {        errWithStack = errors.Errorf("%+v", r)    }    return errWithStack}func main() {    logger := log.New(os.Stdout, "", 0)    defer func() {        if r := recover(); r != nil {            err := handlePanic(r)            logger.Println(                "panic occurred",                "msg", err.Error(),                "stack", fmt.Sprintf("%+v", err),            )        }    }()    fmt.Println(foo())}// 输出:// panic occurred msg: runtime error: invalid memory address or nil pointer dereference// stack: runtime error: invalid memory address or nil pointer dereference// main.handlePanic//        /tmp/sandbox239055659/prog.go:19// main.main.func1...

以上就是今天的内容!recover 函数并不是 Golang 开发者的日常必备工具,但正如你所看到的,它在某些情况下非常有用。SqJ28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-35289-0.html利用 Golang 中的 Recover 处理错误

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: .NET开源的处理分布式事务的解决方案

下一篇: Nginx如何开启GZIP文件压缩,你学会了吗?

标签:
  • 热门焦点
  • 5月安卓手机好评榜:魅族20 Pro夺冠

    性能榜和性价比榜之后,我们来看最后的安卓手机好评榜,数据来源安兔兔评测,收集时间2023年5月1日至5月31日,仅限国内市场。第一名:魅族20 Pro好评率:97.50%不得不感慨魅族老品牌还
  • 六大权益!华为8月服务日开启:手机免费贴膜、维修免人工费

    8月5日消息,一年一度的华为开发者大会2023(Together)日前在松山湖拉开帷幕,与此同时,华为8月服务日也式开启,到店可享六大专属权益。华为用户可在华为商城Ap
  • 学习JavaScript的10个理由...

    作者 | Simplilearn编译 | 王瑞平当你决心学习一门语言的时候,很难选择到底应该学习哪一门,常用的语言有Python、Java、JavaScript、C/CPP、PHP、Swift、C#、Ruby、Objective-
  • 十个简单但很有用的Python装饰器

    装饰器(Decorators)是Python中一种强大而灵活的功能,用于修改或增强函数或类的行为。装饰器本质上是一个函数,它接受另一个函数或类作为参数,并返回一个新的函数或类。它们通常用
  • Python异步IO编程的进程/线程通信实现

    这篇文章再讲3种方式,同时讲4中进程间通信的方式一、 Python 中线程间通信的实现方式共享变量共享变量是多个线程可以共同访问的变量。在Python中,可以使用threading模块中的L
  • 2023年,我眼中的字节跳动

    此时此刻(2023年7月),字节跳动从未上市,也从未公布过任何官方的上市计划;但是这并不妨碍它成为中国最受关注的互联网公司之一。从2016-17年的抖音强势崛起,到2018年的“头腾
  • 花7万退货退款无门:谁在纵容淘宝珠宝商家造假?

    来源:极点商业作者:杨铭在淘宝购买珠宝玉石后,因为保证金不够赔付,店铺关闭,退货退款难、维权无门的比比皆是。“提供相关产品鉴定证书,支持全国复检,可以30天无理由退换货。&
  • 华为HarmonyOS 4.0将于8月4日发布 或搭载AI大模型技术

    华为宣布HarmonyOS4.0将于8月4日正式发布。此前,华为已经针对开发者公布了HarmonyOS4.0,以便于开发者提前进行适配,也因此被曝光出了一些新系统的特性
  • 联想的ThinkBook Plus下一版曝光,键盘旁边塞个平板

    ThinkBook Plus 是联想的一个特殊笔记本类别,它在封面放入了一块墨水屏,也给人留下了较为深刻的印象。据有人爆料,联想的下一款 ThinkBook Plus 可能更特殊,它
Top