分享
  1. 首页
  2. 文章

16-Go语言异常处理

极客江南 · · 2715 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

异常处理

  • 程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常
  • golang中提供了两种处理异常的方式
    • 一种是程序发生异常时, 将异常信息反馈给使用者
    • 一种是程序发生异常时, 立刻退出终止程序继续运行

打印异常信息

  • Go语言中提供了两种创建异常信息的方式
  • 方式一: 通过fmt包中的Errorf函数创建错误信息, 然后打印
package main
import "fmt"
func main() {
 // 1.创建错误信息
 var err error = fmt.Errorf("这里是错误信息")
 // 2.打印错误信息
 fmt.Println(err) // 这里是错误信息
}
  • 方式二: 通过errors包中的New函数创建错误信息,然后打印
package main
import "fmt"
func main() {
 // 1.创建错误信息
 var err error = errors.New("这里是错误信息")
 // 2.打印错误信息
 fmt.Println(err) // 这里是错误信息
}
  • 两种创建异常信息实现原理解析
    • Go语言中创建异常信息其实都是通过一个error接口实现的
    • Go语言再builtin包中定义了一个名称叫做error的接口. 源码如下
package builtin
// 定义了一个名称叫做error的接口
// 接口中声明了一个叫做Error() 的方法
type error interface {
 Error() string
}
  • 在errors包中定义了一个名称叫做做errorString的结构体, 利用这个结构体实现了error接口中指定的方法
  • 并且在errors 包中还提供了一个New方法, 用于创建实现了error接口的结构体对象, 并且在创建时就会把指定的字符串传递给这个结构体
// 指定包名为errors
package errors 
// 定义了一个名称叫做errorString的结构体, 里面有一个字符串类型属性s
type errorString struct {
 s string
}
// 实现了error接口中的Error方法
// 内部直接将结构体中保存的字符串返回
func (e *errorString) Error() string {
 return e.s
}
// 定义了一个New函数, 用于创建异常信息
// 注意: New函数的返回值是一个接口类型
func New(text string) error {
 // 返回一个创建好的errorString结构体地址
 return &errorString{text}
}
  • fmt包中Errorf底层的实现原理其实就是在内部自动调用了errors包中的New函数
func Errorf(format string, a ...interface{}) error {
 return errors.New(Sprintf(format, a...))
}
  • 应用场景
package main
import "fmt"
func div(a, b int) (res int, err error) {
 if(b == 0){
 // 一旦传入的除数为0, 就会返回error信息
 err = errors.New("除数不能为0")
 }else{
 res = a / b
 }
 return
}
func main() {
 //res, err := div(10, 5)
 res, err := div(10, 0)
 if(err != nil){
 fmt.Println(err) // 除数不能为0
 }else{
 fmt.Println(res) // 2
 }
}

中断程序

  • Go语言中提供了一个叫做panic函数, 用于发生异常时终止程序继续运行
package main
import "fmt"
func div(a, b int) (res int) {
 if(b == 0){
 //一旦传入的除数为0, 程序就会终止
 panic("除数不能为0")
 }else{
 res = a / b
 }
 return
}
func main() {
 res := div(10, 0)
 fmt.Println(res)
}
  • Go语言中有两种方式可以触发panic终止程序
    • 我们自己手动调用panic函数
    • 程序内部出现问题自动触发panic函数
package main
import "fmt"
func main() {
 // 例如:数组角标越界, 就会自动触发panic
 var arr = [3]int{1, 3, 5}
 arr[5] = 666 // 报错
 fmt.Println(arr)
 // 例如:除数为0, 就会自动触发panic
 var res = 10 / 0
 fmt.Println(res)
}
  • 除非是不可恢复性、导致系统无法正常工作的错误, 否则不建议使用panic

恢复程序

  • 程序和人一样都需要具备一定的容错能力, 学会知错就改. 所以如果不是不可恢复性、导致系统无法正常工作的错误, 如果发生了panic我们需要恢复程序, 让程序继续执行,并且需要记录到底犯了什么错误
  • 在Go语言中我们可以通过defer和recover来实现panic异常的捕获, 让程序继续执行
package main
import "fmt"
func div(a, b int) (res int) {
 // 定义一个延迟调用的函数, 用于捕获panic异常
 // 注意: 一定要在panic之前定义
 defer func() {
 if err := recover(); err != nil{
 res = -1
 fmt.Println(err) // 除数不能为0
 }
 }()
 if(b == 0){
 //err = errors.New("除数不能为0")
 panic("除数不能为0")
 }else{
 res = a / b
 }
 return
}
func setValue(arr []int, index int ,value int) {
 arr[index] = value
}
func main() {
 res := div(10, 0)
 fmt.Println(res) // -1
}
  • panic注意点
    • panic异常会沿着调用堆栈向外传递, 所以也可以在外层捕获
package main
import "fmt"
func div(a, b int) (res int) {
 if(b == 0){
 //err = errors.New("除数不能为0")
 panic("除数不能为0")
 }else{
 res = a / b
 }
 return
}
func main() {
 // panic异常会沿着调用堆栈向外传递, 所以也可以在外层捕获
 defer func() {
 if err := recover(); err != nil{
 fmt.Println(err) // 除数不能为0
 }
 }()
 div(10, 0)
}
  • 多个异常,只有第一个会被捕获
package main
import "fmt"
func test1() {
 // 多个异常,只有第一个会被捕获
 defer func() {
 if err := recover(); err != nil{
 fmt.Println(err) // 异常A
 }
 }()
 panic("异常A") // 相当于return, 后面代码不会继续执行
 panic("异常B")
}
func main() {
 test1(10, 0)
}
  • 如果有异常写在defer中, 那么只有defer中的异常会被捕获
package main
import "fmt"
func test2() {
 // 如果有异常写在defer中, 那么只有defer中的异常会被捕获
 defer func() {
 if err := recover(); err != nil{
 fmt.Println(err) // 异常A
 }
 }()
 panic("异常A")
 defer func() { // defer会在函数执行完毕之后才会执行, 所以会在("异常A")之后执行
 panic("异常B")
 }()
}
func main() {
 test1(10, 0)
}


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:极客江南

查看原文:16-Go语言异常处理

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
2715 次点击
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏