異常處理
- 程序運(yùn)行時(shí),發(fā)生的不被期望的事件,它阻止了程序按照程序員的預(yù)期正常執(zhí)行,這就是異常
- golang中提供了兩種處理異常的方式
- 一種是程序發(fā)生異常時(shí), 將異常信息反饋給使用者
- 一種是程序發(fā)生異常時(shí), 立刻退出終止程序繼續(xù)運(yùn)行
打印異常信息
- Go語(yǔ)言中提供了兩種創(chuàng)建異常信息的方式
- 方式一: 通過fmt包中的Errorf函數(shù)創(chuàng)建錯(cuò)誤信息, 然后打印
package main
import "fmt"
func main() {
// 1.創(chuàng)建錯(cuò)誤信息
var err error = fmt.Errorf("這里是錯(cuò)誤信息")
// 2.打印錯(cuò)誤信息
fmt.Println(err) // 這里是錯(cuò)誤信息
}
- 方式二: 通過errors包中的New函數(shù)創(chuàng)建錯(cuò)誤信息,然后打印
package main
import "fmt"
func main() {
// 1.創(chuàng)建錯(cuò)誤信息
var err error = errors.New("這里是錯(cuò)誤信息")
// 2.打印錯(cuò)誤信息
fmt.Println(err) // 這里是錯(cuò)誤信息
}
- 兩種創(chuàng)建異常信息實(shí)現(xiàn)原理解析
- Go語(yǔ)言中創(chuàng)建異常信息其實(shí)都是通過一個(gè)error接口實(shí)現(xiàn)的
- Go語(yǔ)言再builtin包中定義了一個(gè)名稱叫做error的接口. 源碼如下
package builtin
// 定義了一個(gè)名稱叫做error的接口
// 接口中聲明了一個(gè)叫做Error() 的方法
type error interface {
Error() string
}
- 在errors包中定義了一個(gè)名稱叫做做errorString的結(jié)構(gòu)體, 利用這個(gè)結(jié)構(gòu)體實(shí)現(xiàn)了error接口中指定的方法
- 并且在errors 包中還提供了一個(gè)New方法, 用于創(chuàng)建實(shí)現(xiàn)了error接口的結(jié)構(gòu)體對(duì)象, 并且在創(chuàng)建時(shí)就會(huì)把指定的字符串傳遞給這個(gè)結(jié)構(gòu)體
// 指定包名為errors
package errors
// 定義了一個(gè)名稱叫做errorString的結(jié)構(gòu)體, 里面有一個(gè)字符串類型屬性s
type errorString struct {
s string
}
// 實(shí)現(xiàn)了error接口中的Error方法
// 內(nèi)部直接將結(jié)構(gòu)體中保存的字符串返回
func (e *errorString) Error() string {
return e.s
}
// 定義了一個(gè)New函數(shù), 用于創(chuàng)建異常信息
// 注意: New函數(shù)的返回值是一個(gè)接口類型
func New(text string) error {
// 返回一個(gè)創(chuàng)建好的errorString結(jié)構(gòu)體地址
return errorString{text}
}
- fmt包中Errorf底層的實(shí)現(xiàn)原理其實(shí)就是在內(nèi)部自動(dòng)調(diào)用了errors包中的New函數(shù)
func Errorf(format string, a ...interface{}) error {
return errors.New(Sprintf(format, a...))
}
package main
import "fmt"
func div(a, b int) (res int, err error) {
if(b == 0){
// 一旦傳入的除數(shù)為0, 就會(huì)返回error信息
err = errors.New("除數(shù)不能為0")
}else{
res = a / b
}
return
}
func main() {
//res, err := div(10, 5)
res, err := div(10, 0)
if(err != nil){
fmt.Println(err) // 除數(shù)不能為0
}else{
fmt.Println(res) // 2
}
}
中斷程序
- Go語(yǔ)言中提供了一個(gè)叫做panic函數(shù), 用于發(fā)生異常時(shí)終止程序繼續(xù)運(yùn)行
package main
import "fmt"
func div(a, b int) (res int) {
if(b == 0){
//一旦傳入的除數(shù)為0, 程序就會(huì)終止
panic("除數(shù)不能為0")
}else{
res = a / b
}
return
}
func main() {
res := div(10, 0)
fmt.Println(res)
}
- Go語(yǔ)言中有兩種方式可以觸發(fā)panic終止程序
- 我們自己手動(dòng)調(diào)用panic函數(shù)
- 程序內(nèi)部出現(xiàn)問題自動(dòng)觸發(fā)panic函數(shù)
package main
import "fmt"
func main() {
// 例如:數(shù)組角標(biāo)越界, 就會(huì)自動(dòng)觸發(fā)panic
var arr = [3]int{1, 3, 5}
arr[5] = 666 // 報(bào)錯(cuò)
fmt.Println(arr)
// 例如:除數(shù)為0, 就會(huì)自動(dòng)觸發(fā)panic
var res = 10 / 0
fmt.Println(res)
}
- 除非是不可恢復(fù)性、導(dǎo)致系統(tǒng)無法正常工作的錯(cuò)誤, 否則不建議使用panic
恢復(fù)程序
- 程序和人一樣都需要具備一定的容錯(cuò)能力, 學(xué)會(huì)知錯(cuò)就改. 所以如果不是不可恢復(fù)性、導(dǎo)致系統(tǒng)無法正常工作的錯(cuò)誤, 如果發(fā)生了panic我們需要恢復(fù)程序, 讓程序繼續(xù)執(zhí)行,并且需要記錄到底犯了什么錯(cuò)誤
- 在Go語(yǔ)言中我們可以通過defer和recover來實(shí)現(xiàn)panic異常的捕獲, 讓程序繼續(xù)執(zhí)行
package main
import "fmt"
func div(a, b int) (res int) {
// 定義一個(gè)延遲調(diào)用的函數(shù), 用于捕獲panic異常
// 注意: 一定要在panic之前定義
defer func() {
if err := recover(); err != nil{
res = -1
fmt.Println(err) // 除數(shù)不能為0
}
}()
if(b == 0){
//err = errors.New("除數(shù)不能為0")
panic("除數(shù)不能為0")
}else{
res = a / b
}
return
}
func setValue(arr []int, index int ,value int) {
arr[index] = value
}
func main() {
res := div(10, 0)
fmt.Println(res) // -1
}
- panic注意點(diǎn)
- panic異常會(huì)沿著調(diào)用堆棧向外傳遞, 所以也可以在外層捕獲
package main
import "fmt"
func div(a, b int) (res int) {
if(b == 0){
//err = errors.New("除數(shù)不能為0")
panic("除數(shù)不能為0")
}else{
res = a / b
}
return
}
func main() {
// panic異常會(huì)沿著調(diào)用堆棧向外傳遞, 所以也可以在外層捕獲
defer func() {
if err := recover(); err != nil{
fmt.Println(err) // 除數(shù)不能為0
}
}()
div(10, 0)
}
- 多個(gè)異常,只有第一個(gè)會(huì)被捕獲
package main
import "fmt"
func test1() {
// 多個(gè)異常,只有第一個(gè)會(huì)被捕獲
defer func() {
if err := recover(); err != nil{
fmt.Println(err) // 異常A
}
}()
panic("異常A") // 相當(dāng)于return, 后面代碼不會(huì)繼續(xù)執(zhí)行
panic("異常B")
}
func main() {
test1(10, 0)
}
- 如果有異常寫在defer中, 那么只有defer中的異常會(huì)被捕獲
package main
import "fmt"
func test2() {
// 如果有異常寫在defer中, 并且其它異常寫在defer后面, 那么只有defer中的異常會(huì)被捕獲
defer func() {
if err := recover(); err != nil{
fmt.Println(err) // 異常A
}
}()
defer func() {
panic("異常B")
}()
panic("異常A")
}
func main() {
test1(10, 0)
}
到此這篇關(guān)于Go語(yǔ)言異常處理案例解析的文章就介紹到這了,更多相關(guān)Go語(yǔ)言異常處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
您可能感興趣的文章:- Golang中異常處理機(jī)制詳解
- 深入理解golang的異常處理機(jī)制
- 利用golang實(shí)現(xiàn)封裝trycatch異常處理實(shí)例代碼
- go語(yǔ)言異常panic和恢復(fù)recover用法實(shí)例
- 小學(xué)生也能看懂的Golang異常處理recover panic