[golang note] 错误处理
错误处理
• 错误处理的标准模式
√ golang错误处理的标准模式:error接口。
√ golang函数如果要返回错误,规范上是将error作为多返回值中的最后一个,但这并非是强制要求。
▶ error接口
type error interface {
Error() string
}
▶ 内置的error类型使用
▪ 语法如下
func 函数名(参数列表) (返回值列表, err error) {
// 函数体
}
▪ 错误处理
例如我们有一个这样的函数:
func Foo(param int) (n int, err error) {
// 函数体
}
调用函数时建议按如下方式处理错误:
n, err := Foo()
if err != nil {
// 错误处理
} else {
// 使用返回值n
}
▪ 示例如下
package main import (
"errors"
"fmt"
) func divide(dividend float64, divisor float64) (result float64, err error) {
if divisor == {
return -, errors.New("除数为0")
} return dividend / divisor, nil
} func main() {
result, err := divide(, )
if err != nil {
fmt.Println(err.Error())
} else {
fmt.Println("result =", result)
} result, err = divide(, )
if err != nil {
fmt.Println(err.Error())
} else {
fmt.Println("result =", result)
}
}
▶ 自定义error类型使用
√ golang错误处理支持自定义的error类型,只需要为自定义error类型实现Error接口即可。
▪ 语法如下
type CustomError struct {
...
}
func (e *CustomError) Error() string {
// 函数体
}
▪ 示例如下
package main import (
"fmt"
) type MathError struct {
Op string
info string
} func (e *MathError) Error() string {
return "Math operation " + e.Op + " error : " + e.info
} func divide(dividend float64, divisor float64) (result float64, err error) {
if divisor == {
return -, &MathError{"division", "divisor is zero"}
} return dividend / divisor, nil
} func main() {
result, err := divide(, )
if err != nil {
fmt.Println(err.Error())
} else {
fmt.Println("result =", result)
} result, err = divide(, )
if err != nil {
fmt.Println(err.Error())
} else {
fmt.Println("result =", result)
}
}
▪ 类型转换
√ 如果处理错误时需要获取详细信息,而不仅仅满足于打印一句错误信息,那就需要用到类型转换。
package main import (
"fmt"
) type MathError struct {
Op string
info string
} func (e *MathError) Error() string {
return "Math operation " + e.Op + " error : " + e.info
} func divide(dividend float64, divisor float64) (result float64, err error) {
if divisor == {
return -, &MathError{"division", "divisor is zero"}
} return dividend / divisor, nil
} func main() {
result, err := divide(, )
if err != nil {
// error类型转换为*MathError指针,因为接口定义传入类型对象为*MathError指针
// 如果接口定义时传入类型对象为MathError,那么这里的写法为err.(MathError)
if e, ok := err.(*MathError); ok {
fmt.Println(e.info)
}
} else {
fmt.Println("result =", result)
}
}
资源释放
在c++程序中,经常要注意内存指针、文件句柄、网络套接字等等资源的释放,特别需要注意其释放的时机。而golang使用defer
关键字和背后的内部机制简单地解决了资源释放的问题。
√ defer关键字能保证其后的代码能在函数退出前调用。
√ 一个函数中可以存在多个defer语句,需要注意的是defer语句的调用是遵照先进后出的原则,即最后一个defer语句将最先被执行。
√ 可以在defer后加一个匿名函数来进行复杂的清理工作。
• 简单的清理工作
▶ 语法如下
func 函数名(参数列表) (返回值列表) {
...
// 资源申请
defer 清理函数
...
}
▶ 示例如下
package main import (
"io"
"os"
) func CopyFile(dst, src string) (w int64, err error) {
srcFile, err := os.Open(src)
if err != nil {
return
}
defer srcFile.Close() dstFile, err := os.Create(dst)
if err != nil {
return
}
defer dstFile.Close() return io.Copy(dstFile, srcFile)
} func main() {
CopyFile("D:/2.txt", "D:/1.txt")
}
▶ 先进后出规则
package main import (
"fmt"
) func Test() {
defer fmt.Println()
defer fmt.Println()
defer fmt.Println()
} func main() {
Test()
}
• 复杂的清理工作
▶ 语法如下
func 函数名(参数列表) (返回值列表) {
...
// 资源申请
defer func() {
// 复杂的清理工作
} ()
...
}
▶ 示例如下
package main import (
"fmt"
"io"
"os"
) func CopyFile(dst, src string) (w int64, err error) {
srcFile, err := os.Open(src)
if err != nil {
return
}
defer func() {
fmt.Println("close file :", src)
srcFile.Close()
}() dstFile, err := os.Create(dst)
if err != nil {
return
}
defer func() {
fmt.Println("close file :", dst)
dstFile.Close()
}() return io.Copy(dstFile, srcFile)
} func main() {
CopyFile("D:/2.txt", "D:/1.txt")
}
异常处理
一些高级语言中一般提供类似try...catch...finally...的语法,用于捕获异常。golang提供panic和recover两个关键字用于异常处理。
• panic
panic在golang中是一个内置函数,接收一个interface{}类型的值作为参数:
func panic(interface{}) {
...
}
当一个函数执行过程中调用panic函数时,函数执行流程将立即终止,但panic之前的defer关键字延迟执行的语句将正常执行,之后该函数将返回到上层调用函数,并逐层向上执行panic流程,直至函数所属的goroutine中所有正在执行函数终止。错误信息将被报告,包括在调用panic()函数时传入的参数。下面用一个示例说明:
package main import (
"fmt"
) func MyFunc1() {
defer fmt.Println("MyFunc1 defer 1") panic("MyFunc1 panic test") defer fmt.Println("MyFunc1 defer 2")
} func MyFunc2() {
defer fmt.Println("MyFunc2 defer 1") MyFunc1() defer fmt.Println("MyFunc2 defer 2")
} func main() {
MyFunc2()
}
程序输出如下:

• recover
recover在golang中是一个内置函数,返回一个interface{}类型的值作为参数:
func recover() interface{} {
...
}
panic函数触发后不会立即返回,而是先defer,再返回。如果defer的时候,有办法将panic捕获到,然后及时进行异常处理,并阻止panic传递,那处理机制就完善了。因此golang提供了recover内置函数,用于捕获panic并阻止其向上传递。需要注意的是,recover之后,逻辑并不会恢复到panic处,函数还是会在defer之后返回,但是所属goroutine将不会退出。
▶ 本层函数处理
package main import (
"fmt"
) func MyFunc1() {
defer func() {
fmt.Println("MyFunc1 defer 1")
if r := recover(); r != nil {
fmt.Println("Runtime error caught :", r)
}
}() panic("MyFunc1 panic test") fmt.Println("MyFunc1 defer 2")
} func MyFunc2() {
defer fmt.Println("MyFunc2 defer 1") MyFunc1() defer fmt.Println("MyFunc2 defer 2")
} func main() {
MyFunc2()
}
程序输出如下:

▶ 上层函数处理
package main import (
"fmt"
) func MyFunc1() {
defer fmt.Println("MyFunc1 defer 1") panic("MyFunc1 panic test") fmt.Println("MyFunc1 defer 2")
} func MyFunc2() {
defer func() {
fmt.Println("MyFunc1 defer 2")
if r := recover(); r != nil {
fmt.Println("Runtime error caught :", r)
}
}() MyFunc1() defer fmt.Println("MyFunc2 defer 2")
} func main() {
MyFunc2()
}
程序输出如下:

• 模拟try...catch...语法
▶ 语法如下
func Try(f func(), handler func(interface{})) {
defer func() {
if err := recover(); err != nil {
handler(err)
}
}()
f()
}
▶ 示例如下
package main import (
"fmt"
) func Try(f func(), handler func(interface{})) {
defer func() {
if err := recover(); err != nil {
handler(err)
}
}() f()
} func main() {
Try(func() {
panic("main panic")
}, func(e interface{}) {
fmt.Println(e)
})
}
[golang note] 错误处理的更多相关文章
- 个人犯的一个golang routine错误
这个其实不是错误,2个写法没有区别.-2015.11.22 认识golang也不少时间了,也做过几个项目.最近发现之前用golang写的一个服务,内存涨得比较快,一直没找出来原因来.今天把疑惑发到群里 ...
- [golang note] 环境搭建
LiteIDE(windows) • golang安装 ▶ 下载对应操作系统的版本并安装,下载地址:http://www.golangtc.com/download,譬如这里下载的是go1.6.win ...
- [golang note] 网络编程 - RPC编程
net包 • 官方文档 http://godoc.golangtc.com/pkg/net/ Package net provides a portable interface for network ...
- [golang note] 接口使用
侵入式接口 √ 在其他一些编程语言中,接口主要是作为不同组件之间的契约存在,即规定双方交互的规约. √ 对契约的实现是强制的,即必须确保用户的确实现了该接口,而实现一个接口,需要从该接口继承. √ 如 ...
- [golang note] 匿名组合
匿名组合 golang也提供了继承机制,但采用组合的文法,因此称为匿名组合.与其他语言不同, golang很清晰地展示出类的内存布局是怎样的. • 非指针方式组合 ▶ 基本语法 // 基类 type ...
- [golang note] 函数定义
普通函数定义 √ golang函数基本组成:关键字func.函数名.参数列表.返回值.函数体和返回语句. • 语法如下 func 函数名(参数列表) (返回值列表) { // 函数体 } • 示例如下 ...
- [golang note] 内建类型
基础类型 √ golang内建基础类型有布尔类型.整数类型.浮点类型.复数类型.字符串类型.字符类型和错误类型. 复合类型 √ golang支持的复合类型有指针.数组.数组切片.字典.通道.结构体和接 ...
- [golang note] 数组切片
数组 √ golang数组包含的每个数据称为数组元素(element),数组包含的元素个数被称为数组长度(length). √ golang数组的长度在定义后不可更改,并且在声明时可以是一个常量或常量 ...
- [golang note] 变量常量
变量 • 变量声明 √ golang变量声明的关键字为var. √ golang变量声明时类型信息放置在变量名之后. ▶ 单个变量声明 ▪ 语法如下 var name type ▪ 示例如下 var ...
随机推荐
- 数据库 Oracle数据库对象二
视图 --视图是对表逻辑抽象 --视图的好处:简化查询 --视图是一种虚表 --视图建立在已有表的基础上,视图赖以建立的这些吧称为基表. --向视图提供数据内容的语句为select语句,可以将视图理解 ...
- Android的Parcelable中describeContents方法的作用
这个方法返回的值通常为0,那什么情况下需要填写其他值呢? 这个方法到目前为止返回其他唯一有效的值就是CONTENTS_FILE_DESCRIPTOR(0x01),指明这个Parcel的内容包含文件描述 ...
- 关于js中遍历总结
1.for循环 var arr = []; for (var i = 0; i < arr.length; i++) { if (条件1) return; if (条件2) break; if ...
- 集成学习AdaBoost算法——学习笔记
集成学习 个体学习器1 个体学习器2 个体学习器3 ——> 结合模块 ——>输出(更好的) ... 个体学习器n 通常,类似求平均值,比最差的能好一些,但是会比最好的差. 集成可能提 ...
- [转]ODBC编程指南
DM4 ODBC编程指南本章结合DM4数据库的特点,比较全面系统的介绍ODBC的基本概念以及DM4 ODBC DRIVER的使用方法,以便用户更好地使用DM4 ODBC编写应用程序.ODBC提供给你访 ...
- windows系统添加服务命令
管理员身份进入cmd sc create TestSvr binPath= D:\Program Files\test.exe start= auto
- IDEA破解后无法启动
在网上找了破解IDEA的方法 原文:https://blog.csdn.net/qq_38637558/article/details/78914772 ①到这个地方下载 IntelliJ IDEA ...
- Struts 2再曝远程代码执行漏洞S2-037
导读今年4月份,Apache Stuts 2之上发现的S2-033远程代码执行漏洞,以迅雷不及掩耳之势席卷而来.其利用代码很快就在短时间内迅速传播.而且官方针对这个高危漏洞的修复方案还是无效的. 悲剧 ...
- awk sed grep 详解
Linux的文本处理工具浅谈 awk [功能说明] 用于文本处理的语言(取行,过滤),支持正则 NR代表行数,$n取某一列,$NF最后一列 NR==20,NR==30 从20行到30行 FS竖着切,列 ...
- 如何使用java指令执行含package的class文件
代码文件存放在E:/Temp/JAVA_TEMP/tmp文件夹,代码如下: package tmp; public class Temp { public static void main(Strin ...