一、defer AND return
1、defer时在return之后执行的,但是在写代码的时候,defer要写在return之前,否则defer在return后不会执行;
函数执行了return语句、运行到函数结尾自动返回、对应的goroutine panic
if err!=nil{
return
}
defer func(){
fmt.Println("defer")
}()
//此处return后,defer不会执行;
defer会将待执行函数加入到一个类似栈的地方,等return之后在来执行这个栈里面的函数;所以return之后的defer是没在栈里的;
2、defer和return
defer函数的执行是在return 函数之后才执行;
如果函数的返回值是无名的,则go语言会在执行return的时候会执行一个类似创建一个临时变量作为保存return值的动作;
有名返回值的函数,由于返回值在函数定义的时候已经将该变量进行定义,在执行return的时候会先执行返回值保存操作,而后续的defer函数会改变这个返回值(虽然defer是在return之后执行的,但是由于使用的函数定义的变量,所以执行defer操作后对该变量的修改会影响到return的值)
解决方法:
func f() (r int) {
t := 5
//赋值
r = t
//在return之前,执行defer函数,defer函数没有对返回值r进行修改,只是修改了变量t
func() {
t = t + 5
}
return
}
func f() (r int) {
//给返回值赋值
r = 1
/**
* 这里修改的r是函数形参的值,是外部传进来的
* func(r int){}里边r的作用域只该func内,修改该值不会改变func外的r值
*/
func(r int) {
r = r + 5
}(r)
return
}
3、panic和recover
panic内置函数停止当前goroutine的正常执行,当函数F调用panic时,函数F的正常执行被立即停止,然后运行所有在F函数中的defer函数,然后F返回到调用他的函数对于调用者G,F函数的行为就像panic一样,终止G的执行并运行G中所defer函数,此过程会一直继续执行到goroutine所有的函数。panic可以通过内置的recover来捕获。
recover内置函数用来管理含有panic行为的goroutine,recover运行在defer函数中,获取panic抛出的错误值,并将程序恢复成正常执行的状态。如果在defer函数之外调用recover,那么recover不会停止并且捕获panic错误如果goroutine中没有panic或者捕获的panic的值为nil,recover的返回值也是nil。由此可见,recover的返回值表示当前goroutine是否有panic行为
如果一直没有recover,抛出的panic到当前goroutine最上层函数时,程序直接异常终止
recover都是在当前的goroutine里进行捕获的,这就是说,对于创建goroutine的外层函数,如果goroutine内部发生panic并且内部没有用recover,外层函数是无法用recover来捕获的,这样会造成程序崩溃
ecover返回的是interface{}类型而不是go中的 error 类型
有疑问加站长微信联系(非本文作者)