百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

golang 异常处理(golang 错误处理)

toyiye 2024-07-03 02:01 13 浏览 0 评论

error接口

Go语言引入了一个关于错误处理的标准模式, 即error接口, 它是Go语言内建的接口类型, 该接口的定义如下:

type error interface{
    Error() string
}

Go语言的标准库代码包errors(安装目录/src/errors/errors.go)为用户提供如下方法:

package errors
func New(text string) error {
    return &errorString{text}
}
type errorString struct {
    s string
}
func (e *errorString) Error() string {
    return e.s
}

另一个可以生成error类型值的方法是调用fmt包(安装目录/src/fmt/print.go)中的Errorf函数:

package fmt
import "errors"
func Errorf(format string, a ...interface{}) error {
    return errors.New(Sprintf(format, a...))
}

【实例】

package main
import(
    "fmt"
    "errors"
)
func main(){
    var err1 error = errors.New("a normal err1")
    fmt.Println(err1) //a normal err1

    var err2 error = fmt.Errorf("%s", "a normal err2")
    fmt.Println(err2) //a normal err2
}

函数通常在最后的返回值中返回错误信息:

package main
import(
"fmt"
"errors"
)
func Divide(a, b float64) (result float64, err error) {
    if b == 0{
        result = 0.0
        err = errors.New("runtime error: divide by zero")
        return
    }
    result = a / b
    err = nil
    return
}
func main(){
    r, err := Divide(10.0, 0)
    if err != nil {
        fmt.Println(err) //错误处理:runtimne error: divide by zero
    }else{
        fmt.Println(r)
    }
}

实例:

err.Error() 作用

package main
import (
"errors"
"fmt"
"reflect"
)
func Divide(a, b float64) (result float64, err error) {
    if b == 0 {
        result = 0.0
        err = errors.New("runtime error: divide by zero")
        return
    }
        result = a / b
        err = nil
    return
}
func main() {
    r, err := Divide(10.0, 0)
    if err != nil {
        fmt.Println(err) //错误处理:runtimne error: divide by zero
        fmt.Println(reflect.TypeOf(err)) //*errors.errorString
        fmt.Println(reflect.TypeOf(err.Error())) //string Error()作用是转换为 sring 类型
    } else {
        fmt.Println(r)
    }
}

panic

在通常情况下, 向程序使用方报告错误状态的方式可以是返回一个额外的error类型值。

但是, 当遇到不可恢复的错误状态的时候, 如数组访问越界、空指针引用等, 这些运行时错误会引起painc异常。这时, 上述错误处理方式显然就不适合了。

反过来讲, 在一般情况下, 我们不应通过调用panic函数来报告普通的错误, 而应该只把它作为报告致命错误的一种方式。当某些不应该发生的场景发生时, 我们就应该调用panic。

一般而言, 当panic异常发生时, 程序会中断运行, 并立即执行在该goroutine(可以先理解成线程, 在中被延迟的函数(defer 机制)。

随后, 程序崩溃并输出日志信息。日志信息包括panic value和函数调用的堆栈跟踪信息。

不是所有的panic异常都来自运行时, 直接调用内置的panic函数也会引发panic异常;panic函数接受任何值作为参数。

func panic(v interface{})

调用panic函数引发的panic异常:

package main
import (
    "fmt"
)
func TestA() {
    fmt.Println("func TestA()")
}
func TestB() {
    panic("func TestB(): panic")
}
func TestC() {
    fmt.Println("func TestC()")
}
func main() {
    TestA()
    TestB() //TestB()发生异常,中断程序
    TestC()
}

运行结果:

func TestA()

panic: func TestB(): panic

goroutine 1 [running]:

main.TestB(...)

D:/Go/study/src/study.go:12

main.main()

D:/Go/study/src/study.go:21 +0x45

exit status 2

内置的panic函数引发的panic异常:

package main
import(
"fmt"
)
func TestA() {
    fmt.Println("func TestA()")
}
func TestB(x int) {
    var a [10]int
    a[x] = 222 //x值为11时,数组越界
}
func TestC() {
    fmt.Println("func TestC()")
}
func main() {
    TestA()
    TestB(11) //TestB()发生异常,中断程序
    TestC()
}

运行结果:

func TestA()

panic: runtime error: index out of range

goroutine 1 [running]:

main.TestB(...)

D:/Go/study/src/study.go:13

main.main()

D:/Go/study/src/study.go:22 +0x2c

exit status 2

recover

运行时panic异常一旦被引发就会导致程序崩溃。这当然不是我们愿意看到的, 因为谁也不能保证程序不会发生任何运行时错误。

不过, Go语言为我们提供了专用于"拦截"运行时panic的内建函数——recover。它可以是当前的程序从运行时panic的状态中恢复并重新获得流程控制权。

func recover() interface{}

注意:recover只有在defer调用的函数中有效。

defer func(){

//recover() //拦截错误

fmt.Println(recover()); //打印错误的信息

}() //调用匿名函数,不要忘记()

如果调用了内置函数recover, 并且定义该defer语句的函数发生了panic异常, recover会使程序从panic中恢复, 并返回panic value。

导致panic异常的函数不会继续运行, 但能正常返回。在未发生panic时调用recover, recover会返回nil。

【实例】

package main
import(
    "fmt"
)
func TestA() {
    fmt.Println("func TestA()")
}
func TestB() (err error) {
    defer func() { //在发生异常时,设置恢复
    if x := recover(); x != nil {
    //panic value被附加到错误信息中;
    //并用err变量接收错误信息,返回给调用者。
    err = fmt.Errorf("internal error: %v", x)
    }
    }()
    panic("func TestB(): panic")
}
func TestC() {
    fmt.Println("func TestC()")
}
func main() {
    TestA()
    err := TestB()
    fmt.Println(err)
    TestC()
/*
运行结果:
func TestA()
internal error: func TestB(): panic
func TestC()
*/
}

延迟调用中引发的错误, 可被后续延迟调用捕获, 但仅最后一个错误可被捕获:

package main
import(
"fmt"
)
func test() {
    defer func() {
    fmt.Println(recover())
    }()
    defer func() {
    panic("defer panic") //最后一个执行的被捕获
    }()
    panic("test panic")
}
func main() {
    test()
    //运行结果:defer panic
}

注意: defer 存在着后进先出的特性, 因此第一个错误是: panic("test panic"), 第二个错误是: panic("defer panic")

【实例】

package main
import(
    "fmt"
)
func TestA() {
    fmt.Println("func TestA()")
}
func TestB(x int) {
    defer func(){
    //recover() //拦截错误
    fmt.Println(recover()); //打印错误的信息
    }() //调用匿名函数,不要忘记()
    var a [10]int
    a[x] = 222 //x值为11时,数组越界
}
func TestC() {
    fmt.Println("func TestC()")
}
func main() {
    TestA()
    TestB(11) //TestB()发生异常,中断程序
    TestC()
}

自定义error

package main
import (
    "fmt"
    "os"
    "time"
)
type PathError struct {
    path string
    op string
    createTime string
    message string
}
func (p *PathError) Error() string {
    return fmt.Sprintf("path=%s \nop=%s \ncreateTime=%s \nmessage=%s", p.path,
    p.op, p.createTime, p.message)
}
func Open(filename string) error {
    file, err := os.Open(filename)
    if err != nil {
    return &PathError{
    path: filename,
    op: "read",
    message: err.Error(),
    createTime: fmt.Sprintf("%v", time.Now()),
    }
}
defer file.Close()
    return nil
}
func main() {
    err := Open("/Users/5lmh/Desktop/go/src/test.txt")
    switch v := err.(type) {
    case *PathError:
    fmt.Println("get path error,", v)
    default:
    }
}

输出结果:

get path error, path=/Users/pprof/Desktop/go/src/test.txt

op=read

createTime=2018-04-05 11:25:17.331915 +0800 CST m=+0.000441790

message=open /Users/pprof/Desktop/go/src/test.txt: no such file or directory

golang recover捕获不到当前函数中的panic异常

package main
import(
"fmt"
"bytes"
"log"
)
func main(){
    var buff bytes.Buffer
    // 自定义一个日志对象
    // 默认的日志写入到buff中
    myLog := log.New(&buff, "", log.LstdFlags)
    // 写入日志
    myLog.Println("line 1")
    myLog.Printf("line %d", 2)
    myLog.Panic("log panic")

    defer func(){
    if err := recover(); err != nil{
    fmt.Println("err: ",err)
    }
    }()
}

原因是defer执行的函数在panic之后才被调用, 也就是说defer根本就没有运行, 因为myLog.Panic已经终止了函数调用, 正确的做法应该是将defer执行的函数移到前面

defer执行的函数需要在panic之前调用

package main
import(
"fmt"
"bytes"
"log"
)
func main(){
    defer func(){
    if err := recover(); err != nil{
        fmt.Println("err: ",err)
    }
    }()
    var buff bytes.Buffer
    // 自定义一个日志对象
    // 默认的日志写入到buff中
    myLog := log.New(&buff, "", log.LstdFlags)
    // 写入日志
    myLog.Println("line 1")
    myLog.Printf("line %d", 2)
    myLog.Panic("log panic")
}

结论: 当然还有一种情况也会捕获不到, 那就是发生异常的代码在另外一个协程, 调用它的函数无法捕获到其中的异常, 这种情况需要在协程中进行捕获;


相关推荐

为何越来越多的编程语言使用JSON(为什么编程)

JSON是JavascriptObjectNotation的缩写,意思是Javascript对象表示法,是一种易于人类阅读和对编程友好的文本数据传递方法,是JavaScript语言规范定义的一个子...

何时在数据库中使用 JSON(数据库用json格式存储)

在本文中,您将了解何时应考虑将JSON数据类型添加到表中以及何时应避免使用它们。每天?分享?最新?软件?开发?,Devops,敏捷?,测试?以及?项目?管理?最新?,最热门?的?文章?,每天?花?...

MySQL 从零开始:05 数据类型(mysql数据类型有哪些,并举例)

前面的讲解中已经接触到了表的创建,表的创建是对字段的声明,比如:上述语句声明了字段的名称、类型、所占空间、默认值和是否可以为空等信息。其中的int、varchar、char和decimal都...

JSON对象花样进阶(json格式对象)

一、引言在现代Web开发中,JSON(JavaScriptObjectNotation)已经成为数据交换的标准格式。无论是从前端向后端发送数据,还是从后端接收数据,JSON都是不可或缺的一部分。...

深入理解 JSON 和 Form-data(json和formdata提交区别)

在讨论现代网络开发与API设计的语境下,理解客户端和服务器间如何有效且可靠地交换数据变得尤为关键。这里,特别值得关注的是两种主流数据格式:...

JSON 语法(json 语法 priority)

JSON语法是JavaScript语法的子集。JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔花括号保存对象方括号保存数组JS...

JSON语法详解(json的语法规则)

JSON语法规则JSON语法是JavaScript对象表示法语法的子集。数据在名称/值对中数据由逗号分隔大括号保存对象中括号保存数组注意:json的key是字符串,且必须是双引号,不能是单引号...

MySQL JSON数据类型操作(mysql的json)

概述mysql自5.7.8版本开始,就支持了json结构的数据存储和查询,这表明了mysql也在不断的学习和增加nosql数据库的有点。但mysql毕竟是关系型数据库,在处理json这种非结构化的数据...

JSON的数据模式(json数据格式示例)

像XML模式一样,JSON数据格式也有Schema,这是一个基于JSON格式的规范。JSON模式也以JSON格式编写。它用于验证JSON数据。JSON模式示例以下代码显示了基本的JSON模式。{"...

前端学习——JSON格式详解(后端json格式)

JSON(JavaScriptObjectNotation)是一种轻量级的数据交换格式。易于人阅读和编写。同时也易于机器解析和生成。它基于JavaScriptProgrammingLa...

什么是 JSON:详解 JSON 及其优势(什么叫json)

现在程序员还有谁不知道JSON吗?无论对于前端还是后端,JSON都是一种常见的数据格式。那么JSON到底是什么呢?JSON的定义...

PostgreSQL JSON 类型:处理结构化数据

PostgreSQL提供JSON类型,以存储结构化数据。JSON是一种开放的数据格式,可用于存储各种类型的值。什么是JSON类型?JSON类型表示JSON(JavaScriptO...

JavaScript:JSON、三种包装类(javascript 包)

JOSN:我们希望可以将一个对象在不同的语言中进行传递,以达到通信的目的,最佳方式就是将一个对象转换为字符串的形式JSON(JavaScriptObjectNotation)-JS的对象表示法...

Python数据分析 只要1分钟 教你玩转JSON 全程干货

Json简介:Json,全名JavaScriptObjectNotation,JSON(JavaScriptObjectNotation(记号、标记))是一种轻量级的数据交换格式。它基于J...

比较一下JSON与XML两种数据格式?(json和xml哪个好)

JSON(JavaScriptObjectNotation)和XML(eXtensibleMarkupLanguage)是在日常开发中比较常用的两种数据格式,它们主要的作用就是用来进行数据的传...

取消回复欢迎 发表评论:

请填写验证码