GO语言
01GO基础-001GO语言简介
01GO基础-002语言环境安装
01GO基础-003Go 语言结构
01GO基础-004Go 语言基础语法1
01GO基础-004Go 语言基础语法2
01GO基础-004Go 语言基础语法3
01GO基础-005Go 语言数据类型
01GO基础-006Go 语言变量
01GO基础-007Go 语言常量
01GO基础-008Go 语言运算符
01GO基础-009条件语句
01GO基础-010循环语句
01GO基础-011函数
01GO基础-012变量作用域
01GO基础-013数组
01GO基础-014指针
01GO基础-015结构体
01GO基础-016切片
01GO基础-017范围(Range)
01GO基础-018Map
01GO基础-019递归函数
01GO基础-020类型转换
01GO基础-021接口
01GO基础-022异常处理
01GO基础-023并发
01GO基础-024strings
01GO基础-025可变参数
01GO基础-026接口2
01GO基础-027异常处理2
01GO基础-028sync包详解
01GO基础-029Context
02GO进阶001包
02GO进阶002init()函数
02GO进阶003包的注意点
02GO进阶003使用go module导入本地包
02GO进阶004 time包
02GO进阶005 file操作
02GO进阶006 io操作
02GO进阶007 os包(文件 I/O、文件属性、目录与链接、创建和移除链接)
02GO进阶008复制文件
02GO进阶009断点续传
02GO进阶010 bufio包
02GO进阶011ioutil包
02GO进阶012遍历文件夹
02GO进阶013并发编程介绍
02GO进阶014Goroutine协程
02GO进阶015 GPM
02GO进阶016 runtime包
02GO进阶017 Channel
02GO进阶018 Goroutine池
02GO进阶019 定时器
02GO进阶020 select
02GO进阶021并发安全和锁
02GO进阶022sync
02GO进阶023原子操作
02GO进阶024 GMP原理与调度
02GO进阶025爬虫小案例
02GO进阶026 面向对象-匿名字段
02GO进阶026 面向对象-接口
02GO进阶027网络编程-互联网协议介绍
02GO进阶027网络编程-socket
02GO进阶027网络编程-http编程
02GO进阶027网络编程-websocket编程
02GO进阶028数据操作-MYSQL
02GO进阶028数据操作-REDIS
02GO进阶028数据操作-RTCD
02GO进阶028数据操作-ZOOKEEPER
02GO进阶028数据操作-KAFKA
02GO进阶028数据操作-RabbitMQ
02GO进阶028数据操作-ElasticSearch
02GO进阶028数据操作-NSQ
02GO进阶028数据操作-memcached
02GO进阶028数据操作-GORM
02GO进阶029beego框架-安装
02GO进阶029beego框架-快速入门
02GO进阶029beego框架-MVC架构介绍-controller设计-参数配置
02GO进阶029beego框架-MVC架构介绍-controller设计-路由设置
02GO进阶029beego框架-MVC架构介绍-controller设计-控制器函数
02GO进阶029beego框架-MVC架构介绍-controller设计-XSRF过滤
02GO进阶029beego框架-MVC架构介绍-controller设计-请求数据处理
02GO进阶029beego框架-MVC架构介绍-controller设计-Session控制
02GO进阶029beego框架-MVC架构介绍-controller设计-过滤器
02GO进阶029beego框架-MVC架构介绍-controller设计-Flash数据
02GO进阶029beego框架-MVC架构介绍-controller设计-URL构建
02GO进阶029beego框架-MVC架构介绍-controller设计-多种格式数据输出
02GO进阶029beego框架-MVC架构介绍-controller设计-表单数据验证
02GO进阶029beego框架-MVC架构介绍-controller设计-错误处理
02GO进阶029beego框架-MVC架构介绍-controller设计-日志处理
02GO进阶029beego框架-MVC架构介绍-model设计-概述
02GO进阶029beego框架-MVC架构介绍-model设计-CRUD操作
02GO进阶029beego框架-MVC架构介绍-model设计-高级查询
02GO进阶029beego框架-MVC架构介绍-model设计-原生SQL查询
02GO进阶029beego框架-MVC架构介绍-model设计-构造查询
02GO进阶029beego框架-MVC架构介绍-model设计-事务处理
02GO进阶029beego框架-MVC架构介绍-model设计-模型定义
02GO进阶029beego框架-MVC架构介绍-model设计-命令模式
02GO进阶029beego框架-MVC架构介绍-model设计-测试用例
02GO进阶029beego框架-MVC架构介绍-view设计-beego 模板语法指南
02GO进阶029beego框架-MVC架构介绍-view设计-模板处理
02GO进阶029beego框架-MVC架构介绍-view设计-其他
本文档使用 MrDoc 发布
-
+
首页
01GO基础-022异常处理
## 错误处理 Go 语言通过内置的错误接口提供了非常简单的错误处理机制。 Go 语言的错误处理采用显式返回错误的方式,而非传统的异常处理机制。这种设计使代码逻辑更清晰,便于开发者在编译时或运行时明确处理错误。 Go 的错误处理主要围绕以下机制展开: 1)error 接口:标准的错误表示。 2)显式返回值:通过函数的返回值返回错误。 3)自定义错误:可以通过标准库或自定义的方式创建错误。 4)panic 和 recover:处理不可恢复的严重错误。 ## error 接口 Go 标准库定义了一个 error 接口,表示一个错误的抽象。 error 类型是一个接口类型,这是它的定义: ```go type error interface { Error() string } ``` 实现 error 接口:任何实现了 Error() 方法的类型都可以作为错误。 Error() 方法返回一个描述错误的字符串。 ## 使用 errors 包创建错误 我们可以在编码中通过实现 error 接口类型来生成错误信息。 创建一个简单错误: 实例 ```go package main import ( "errors" "fmt" ) func main() { err := errors.New("this is an error") fmt.Println(err) // 输出:this is an error } ``` 函数通常在最后的返回值中返回错误信息,使用 errors.New 可返回一个错误信息: ```go func Sqrt(f float64) (float64, error) { if f < 0 { return 0, errors.New("math: square root of negative number") } // 实现 } ``` 在下面的例子中,我们在调用 Sqrt 的时候传递的一个负数,然后就得到了 non-nil 的 error 对象,将此对象与 nil 比较,结果为 true,所以 fmt.Println(fmt 包在处理 error 时会调用 Error 方法)被调用,以输出错误,请看下面调用的示例代码: ```go result, err:= Sqrt(-1) if err != nil { fmt.Println(err) } ``` ## 显式返回错误 Go 中,错误通常作为函数的返回值返回,开发者需要显式检查并处理。 显式返回错误: 实例 ```go package main import ( "errors" "fmt" ) func divide(a, b int) (int, error) { if b == 0 { return 0, errors.New("division by zero") } return a / b, nil } func main() { result, err := divide(10, 0) if err != nil { fmt.Println("Error:", err) } else { fmt.Println("Result:", result) } } ``` 输出: Error: division by zero ## 自定义错误 通过定义自定义类型,可以扩展 error 接口。 自定义错误类型: 实例 ```go package main import ( "fmt" ) type DivideError struct { Dividend int Divisor int } func (e *DivideError) Error() string { return fmt.Sprintf("cannot divide %d by %d", e.Dividend, e.Divisor) } func divide(a, b int) (int, error) { if b == 0 { return 0, &DivideError{Dividend: a, Divisor: b} } return a / b, nil } func main() { _, err := divide(10, 0) if err != nil { fmt.Println(err) // 输出:cannot divide 10 by 0 } } ``` ## fmt 包与错误格式化 fmt 包提供了对错误的格式化输出支持: %v:默认格式。 %+v:如果支持,显示详细的错误信息。 %s:作为字符串输出。 实例 ```go package main import ( "fmt" ) // 定义一个 DivideError 结构 type DivideError struct { dividee int divider int } // 实现 `error` 接口 func (de *DivideError) Error() string { strFormat := ` Cannot proceed, the divider is zero. dividee: %d divider: 0 ` return fmt.Sprintf(strFormat, de.dividee) } // 定义 `int` 类型除法运算的函数 func Divide(varDividee int, varDivider int) (result int, errorMsg string) { if varDivider == 0 { dData := DivideError{ dividee: varDividee, divider: varDivider, } errorMsg = dData.Error() return } else { return varDividee / varDivider, "" } } func main() { // 正常情况 if result, errorMsg := Divide(100, 10); errorMsg == "" { fmt.Println("100/10 = ", result) } // 当除数为零的时候会返回错误信息 if _, errorMsg := Divide(100, 0); errorMsg != "" { fmt.Println("errorMsg is: ", errorMsg) } } ``` 执行以上程序,输出结果为: 100/10 = 10 errorMsg is: Cannot proceed, the divider is zero. dividee: 100 divider: 0 ## 使用 errors.Is 和 errors.As 从 Go 1.13 开始,errors 包引入了 errors.Is 和 errors.As 用于处理错误链: errors.Is 检查某个错误是否是特定错误或由该错误包装而成。 实例 ```go package main import ( "errors" "fmt" ) var ErrNotFound = errors.New("not found") func findItem(id int) error { return fmt.Errorf("database error: %w", ErrNotFound) } func main() { err := findItem(1) if errors.Is(err, ErrNotFound) { fmt.Println("Item not found") } else { fmt.Println("Other error:", err) } } ``` errors.As 将错误转换为特定类型以便进一步处理。 实例 ```go package main import ( "errors" "fmt" ) type MyError struct { Code int Msg string } func (e *MyError) Error() string { return fmt.Sprintf("Code: %d, Msg: %s", e.Code, e.Msg) } func getError() error { return &MyError{Code: 404, Msg: "Not Found"} } func main() { err := getError() var myErr *MyError if errors.As(err, &myErr) { fmt.Printf("Custom error - Code: %d, Msg: %s\n", myErr.Code, myErr.Msg) } } ``` ## panic 和 recover Go 的 panic 用于处理不可恢复的错误,recover 用于从 panic 中恢复。 panic: 导致程序崩溃并输出堆栈信息。 常用于程序无法继续运行的情况。 recover: 捕获 panic,避免程序崩溃。 实例 ```go package main import "fmt" func safeFunction() { defer func() { if r := recover(); r != nil { fmt.Println("Recovered from panic:", r) } }() panic("something went wrong") } func main() { fmt.Println("Starting program...") safeFunction() fmt.Println("Program continued after panic") } ``` 执行以上代码,输出结果为: Starting program... Recovered from panic: something went wrong Program continued after panic
admin
2024年12月20日 14:28
转发文档
收藏文档
上一篇
下一篇
手机扫码
复制链接
手机扫一扫转发分享
复制链接
Markdown文件
分享
链接
类型
密码
更新密码