golang-panic和recover

这两个内置函数用来处理go的运行时错误,panic用来主动抛出错误,recover用来捕获错误。

一、发生panic后,函数会从调用panic的位置返回,逐层向上执行函数的defer语句,直到被recover捕获或运行到最外层函数退出。

package main

import "fmt"
func f3(){
	panic("aaa")
}
func f4(){
	f3()
}
func main(){
	defer func(){
		fmt.Println("a")
		fmt.Println(recover())
	}()
	f4()

}

f3()中的panic向上经过f4(),在main函数的defer中被捕获,执行结果:
在这里插入图片描述
二、recover()只有在defer后面的函数体内被直接调用才能捕获panic异常

//无效
//(1)
recover()
//(2)
defer recover()
//(3)
defer fmt.Println(recover())
//(4)嵌套两层函数,无效
defer func(){
	func(){
		recover()
	}
}
//有效
//(1)
defer func(){
		fmt.Println("a")
		fmt.Println(recover())
	}()
//(2)
func f1(){
	fmt.Println(recover())
}
defer func(){
		fmt.Println("a")
		f1()	
	}()

三、可以有多个panic被抛出(只能出现在延迟调用里面,且只有最后一个panic能被捕获)

package main

import "fmt"

func main(){
	defer func(){
		fmt.Println("a")
		fmt.Println(recover())
	}()
	defer func(){
		fmt.Println("b")
		fmt.Println(recover())
	}()
	defer func(){
		panic("aaa")
	}()
	defer func(){
		panic("bbb")
	}()
	defer func(){
		panic("ccc")
	}()
}

执行结果:
在这里插入图片描述根据defer先进后出的规则只有

	defer func(){
		panic("aaa")
	}()

的异常被捕获,且只有

	defer func(){
		fmt.Println("b")
		fmt.Println(recover())
	}()

捕获到异常。

四、包含init()函数引发的panic只能在init函数中捕获,无法在main函数中被捕获,因为init函数优先于main函数

package main

import "fmt"
func f5(){
	panic("aaa")
}
func init(){
	defer func(){
		fmt.Println(recover())
	}()
	f5()
}
func main(){
	
}

五、函数不能捕获内部协程所抛出的panic

package main

import "fmt"

func f5(){
	panic("aaa")
}
func main(){
	defer func(){
		fmt.Println(recover())
	}()
	go f5()
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值