Go语言学习之路第8天(异常处理)
2020-12-13 05:19
标签:错误 Go语言 host hosts 返回值 完成后 span == invoke 所谓的异常:当GO检测到一个错误时,程序就无法继续执行了,反而出现了一些错误的提示,这就是所谓的"异常"。 所以为了保证程序的健壮性,要对异常的信息进行处理。例如,如下程序,定义一个函数实现整除操作,这个程序对大家来说已经很简单了,实现如下: 但是,大家仔细考虑一下,该方法是否有问题? 如果b的值为0,会出现什么情况? 程序会出现以下的异常信息: 并且整个程序停止运行。 那么出现这种情况,应该怎样进行处理呢?这时就要用到异常处理方法的内容。 Go语言引入了一个关于错误处理的标准模式,即error接口,它是Go语言内建的接口类型,该接口的定义如下: Go语言的标准库代码包errors为用户提供如下方法: 通过以上代码,可以发现error接口的使用是非常简单的(error是一个接口,该接口只声明了一个方法Error(),返回值是string类型,用以描述错误)。下面看一下基本使用, 首先导包: 然后调用其对应的方法: 当然fmt包中也封装了一个专门输出错误信息的方法,如下所示: 了解完基本的语法以后,接下来使用error接口解决Test( )函数被0整除的问题,如下所示: 在Test( )函数中,判断变量b的取值,如果有误,返回错误信息。并且在main( )中接收返回的错误信息,并打印出来。 这种用法是非常常见的,例如,后面讲解到文件操作时,涉及到文件的打开,如下: 在打开文件时,如果文件不存在,或者文件在磁盘上存储的路径写错了,都会出现异常,这时可以使用error记录相应的错误信息。 error返回的是一般性的错误,但是panic函数返回的是让程序崩溃的错误。也就是当遇到不可恢复的错误状态的时候,如数组访问越界、空指针引用等,这些运行时错误会引起painc异常,在一般情况下,我们不应通过调用panic函数来报告普通的错误,而应该只把它作为报告致命错误的一种方式。当某些不应该发生的场景发生时,我们就应该调用panic。 一般而言,当panic异常发生时,程序会中断运行。随后,程序崩溃并输出日志信息。日志信息包括panic value和函数调用的堆栈跟踪信息。 当然,如果直接调用内置的panic函数也会引发panic异常;panic函数接受任何值作为参数。 下面给大家演示一下,直接调用panic函数,是否会导致程序的崩溃。 错误信息如下: 所以,我们在实际的开发过程中并不会直接调用panic( )函数,但是当我们编程的程序遇到致命错误时,系统会自动调用该函数来终止整个程序的运行,也就是系统内置了panic函数。 下面给大家演示一个数组下标越界的问题: 错误信息如下: 通过观察错误信息,发现确实是panic异常,导致了整个程序崩溃。 (1)defer的基本使用 函数定义完成后,只有调用函数才能够执行,并且一经调用立即执行。例如: 先输出"hello",再输出"world"。 但是关键字defer ?于延迟一个函数(或者当前所创建的匿名函数)的执行(注意,defer语句只能出现在函数的内部)。将一个方法延迟到包裹该方法的方法返回时执行,在实际应用中,defer语句可以充当其他语言中try…catch…的角色,也可以用来处理关闭文件句柄等收尾操作。 基本用法如下: 以上两行代码,输出的结果为,先输出"world",再输出"hello"。 (2)defer触发时机 Go官方文档中对defer的执行时机做了阐述,分别是。 (3)defer的执行顺序 一个方法中有多个defer时, defer会将要延迟执行的方法“压栈”,当defer被触发时,将所有“压栈”的方法“出栈”并执行。所以defer的执行顺序是LIFO(先进后出)的。 所以下面这段代码的输出不是1 2 3,而是3 2 1。 (4)defer与return,函数返回值之间的顺序 先说结论:return最先执行->return负责将结果写入返回值中->接着defer开始执行一些收尾工作->最后函数携带当前返回值退出 返回值的表达方式,我们知道根据是否提前声明有两种方式:一种是func test() int 另一种是 func test() (i int),所以两种情况都来说说: 先看一下:func test() int 结果如下: 上面函数Test的返回值属于匿名返回值,返回值在return的时候才确定下来是i的值,具体过程如下: 在这种情况下,defer中的修改是对i执行的,而不是retValue,所以defer返回的依然是retValue。 再看一下:func test() (i int) 结果如下: 这里的函数Test的返回值属于命名返回值,在命名返回值方法中,由于返回值在方法定义时已经被定义,所以没有创建retValue的过程,i就是retValue,defer对于i的修改也会被直接返回。 (5)defer的定义和执行是两个步骤 先说结论:会先将defer后函数的参数部分的值(或者地址)给先记下来【你可以理解为()里头参数的值的会先确定】,后面函数执行完,才会执行defer后函数的{}中的逻辑。 结果如下: (6)尽量避免在for循环中使用defer 看下面的代码: 这是一个循环可打开文件的函数(文件操作之后讲到),defer在紧邻创建资源的语句后声明,看上去逻辑没有什么问题。但是和直接调用相比,defer的执行存在着额外的开销,例如defer会对其后需要的参数进行内存拷贝,还需要对defer结构进行压栈出栈操作。所以在循环中定义defer可能导致大量的资源开销,在本例中,可以将f.Close()语句前的defer去掉,来减少大量defer导致的额外资源消耗。 (7)判断执行没有err之后,再defer释放资源 一些获取资源的操作可能会返回err参数,我们可以选择忽略返回的err参数,但是如果要使用defer进行延迟释放的的话,需要在使用defer之前先判断是否存在err,如果资源没有获取成功,即没有必要也不应该再对资源执行释放操作。如果不判断获取资源是否成功就执行释放操作的话,还有可能导致释放方法执行错误。 正确写法如下。 (8)调用os.Exit时defer不会被执行 上面的defer并不会输出。 运行时panic异常一旦被引发就会导致程序崩溃。这当然不是我们愿意看到的,但谁也不能保证程序不会发生任何运行时错误。 Go语言为我们提供了专用于"拦截"运行时panic的内建函数——recover。它可以让当前的程序从运行时panic的状态中恢复并重新获得流程控制权。 语法如下: 注意:recover只有在defer调用的函数中才有效 示例如下: 结果如下: 通过以上程序,我们发现虽然TestB( )函数会导致整个应用程序崩溃,但是由于在该函数中调用了recover( )函数,所以整个函数并没有崩溃。虽然程序没有崩溃,但是我们也没有看到任何的提示信息,那么怎样才能够看到相应的提示信息呢? 可以直接打印recover( )函数的返回结果,如下所示: 结果如下: 从输出结果发现,确实打印出了相应的错误信息。 但是,如果程序没有出错,也就是数组下标没有越界,会出现什么情况呢? 结果如下: 这时输出的是空,但是我们希望程序没有错误的时候,不输出任何内容。 所以,程序修改如下: 通过以上代码,发现其实就是加了一层判断。 最后还要注意:recover一定要放在可能会发生异常的代码段前面。 Go语言学习之路第8天(异常处理) 标签:错误 Go语言 host hosts 返回值 完成后 span == invoke 原文地址:https://www.cnblogs.com/dacaigouzi1993/p/11129983.html一.异常处理
func Test(a, b int) int {
var result int
result = a / b
return result
}
panic: runtime error: integer divide by zero
1.1 error接口
type error interface {
Error() string
}
package errors
// New returns an error that formats as the given text.
func New(text string) error {
return &errorString{text}
}
// errorString is a trivial implementation of error.
type errorString struct {
s string
}
func (e *errorString) Error() string {
return e.s
}
import "errors"
func main() {
err := errors.New("This is a normal err")
fmt.Println("err:", err.Error())
}
err := fmt.Errorf("%s", "This is a normal err")
fmt.Println("err:", err)
func Test(a, b int) (result int, err error) {
err = nil
if b == 0 {
err = errors.New("除数不能为0")
} else {
result = a / b
}
return result, err
}
func main() {
res, err := Test(10, 0)
if err != nil {
fmt.Println("err:", err)
} else {
fmt.Println(res)
}
}
func Open(name string) (*File, error)
1.2 panic函数
func TestA() {
fmt.Println("func TestA()")
}
func TestB() {
panic("func TestB(): panic")
}
func TestC() {
fmt.Println("func TestcC()")
}
func main() {
TestA()
TestB() //TestB()发生异常,中断程序
TestC()
}
panic: func TestB(): panic
func TestA()
goroutine 1 [running]:
main.TestB(...)
/Users/guanyuji/go/src/awesomeProject/Go基础班第8天/01.go:45
main.main()
/Users/guanyuji/go/src/awesomeProject/Go基础班第8天/01.go:54 +0x98
Process finished with exit code 2
func TestA() {
fmt.Println("func TestA()")
}
func TestB(x int) {
var a [10]int
a[x] = 222
}
func TestC() {
fmt.Println("func TestcC()")
}
func main() {
TestA()
TestB(11)
TestC()
}
func TestA()
panic: runtime error: index out of range
goroutine 1 [running]:
main.TestB(...)
/Users/guanyuji/go/src/awesomeProject/Go基础班第8天/01.go:46
main.main()
/Users/guanyuji/go/src/awesomeProject/Go基础班第8天/01.go:55 +0x7d
Process finished with exit code 2
1.3 延迟调用defer
fmt.Println("hello")
fmt.Println("world")
defer fmt.Println("hello")
fmt.Println("world")
A "defer" statement invokes a function whose execution is deferred to the moment the surrounding function returns, either because the surrounding function executed a return statement, reached the end of its function body, or because the corresponding goroutine is panicking.
func main() {
defer func() {
fmt.Println(1)
}()
defer func() {
fmt.Println(2)
}()
defer func() {
fmt.Println(3)
}()
}
func Test() int {
i := 0
defer func() {
i++
fmt.Println("defer2的值:", i)
}()
defer func() {
i++
fmt.Println("defer1的值:", i)
}()
return i
}
func main() {
fmt.Println("main:", Test())
}
defer1的值: 1
defer2的值: 2
main: 0
func Test() (i int) {
defer func() {
i++
fmt.Println("defer2的值:", i)
}()
defer func() {
i++
fmt.Println("defer1的值:", i)
}()
return i
}
func main() {
fmt.Println("main:", Test())
}
defer1的值: 1
defer2的值: 2
main: 2
func test(i int) int {
return i
}
func main() {
var i int = 1
//defer定义时候test(i)的值就已经确定了,是1,之后就不会变了
defer fmt.Println("i1 = ",test(i))
i++
//defer定义时候test(i)的值就已经确定了,是2,之后就不会变了
defer fmt.Println("i2 = ",test(i))
//defer在定义的时候,i就已经确定了是一个指针类型,地址上的值变了,这里跟着变,是2
defer func(i *int) {
fmt.Println("i3 = ",*i)
}(&i)
//defer在定义的时候,i就已经确定了,是2,之后就不会变了
defer func(i int) {
fmt.Println("i4 = ",i)
}(i)
defer func() {
// 地址,所以后续跟着变
var c = &i
fmt.Println("i5 =" , *c)
}()
// 执行了 i=11 后才调用,此时i值已是11
defer func() {
fmt.Println("i6 =" , i)
}()
i = 11
i6 = 11
i5 = 11
i4 = 2
i3 = 11
i2 = 2
i1 = 1
func DeferTest() {
for i := 0;i
func main() {
fr,err := os.Open("/etc/hosts")
if err != nil{
fmt.Println("os.Open err",err)
return
}
defer fr.Close()
}
unc DeferTest() {
defer func() {
fmt.Println("defer")
}()
os.Exit(0)
}
func main() {
DeferTest()
}
1.4 recover
func recover() interface{}
func TestA() {
fmt.Println("TestA")
}
func TestB(i int) {
//设置recover
defer func() {
recover() //防止程序崩溃
}() //()一定要加上,调用此匿名函数
var a [10]int
a[i] = 222 //当i=11时,会造成数组下标越界,引起panic
}
func TestC() {
fmt.Println("TestC")
}
func main() {
TestA()
TestB(11)
TestC()
}
TestA
TestC
func TestA() {
fmt.Println("TestA")
}
func TestB(i int) {
//设置recover
defer func() {
fmt.Println(recover()) //防止程序崩溃
}() //()一定要加上,调用此匿名函数
var a [10]int
a[i] = 222 //当i=11时,会造成数组下标越界,引起panic
}
func TestC() {
fmt.Println("TestC")
}
func main() {
TestA()
TestB(11)
TestC()
}
TestA
runtime error: index out of range
TestC
func TestA() {
fmt.Println("TestA")
}
func TestB(i int) {
//设置recover
defer func() {
fmt.Println(recover()) //防止程序崩溃
}() //()一定要加上,调用此匿名函数
var a [10]int
a[i] = 222 //当i=11时,会造成数组下标越界,引起panic
}
func TestC() {
fmt.Println("TestC")
}
func main() {
TestA()
TestB(1) //没有越界
TestC()
}
TestA
func TestA() {
fmt.Println("TestA")
}
func TestB(i int) {
//设置recover
defer func() {
//判断是否出现错误
if err := recover();err != nil{
fmt.Println(err) //防止程序崩溃
}
}() //()一定要加上,调用此匿名函数
var a [10]int
a[i] = 222 //当i=11时,会造成数组下标越界,引起panic
}
func TestC() {
fmt.Println("TestC")
}
func main() {
TestA()
TestB(11) //没有越界
TestC()
}