您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“Go語(yǔ)言異常處理的示例分析”,內(nèi)容簡(jiǎn)而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“Go語(yǔ)言異常處理的示例分析”這篇文章吧。
程序運(yùn)行時(shí),發(fā)生的不被期望的事件,它阻止了程序按照程序員的預(yù)期正常執(zhí)行,這就是異常
golang中提供了兩種處理異常的方式
一種是程序發(fā)生異常時(shí), 將異常信息反饋給使用者
一種是程序發(fā)生異常時(shí), 立刻退出終止程序繼續(xù)運(yùn)行
Go語(yǔ)言中提供了兩種創(chuàng)建異常信息的方式
方式一: 通過(guò)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ò)誤信息 }
方式二: 通過(guò)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í)都是通過(guò)一個(gè)error接口實(shí)現(xiàn)的
Go語(yǔ)言再builtin包中定義了一個(gè)名稱(chēng)叫做error的接口. 源碼如下
package builtin // 定義了一個(gè)名稱(chēng)叫做error的接口 // 接口中聲明了一個(gè)叫做Error() 的方法 type error interface { Error() string }
在errors包中定義了一個(gè)名稱(chēng)叫做做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è)名稱(chēng)叫做errorString的結(jié)構(gòu)體, 里面有一個(gè)字符串類(lèi)型屬性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è)接口類(lèi)型 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...)) }
應(yīng)用場(chǎng)景
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)問(wè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)無(wú)法正常工作的錯(cuò)誤, 否則不建議使用panic
程序和人一樣都需要具備一定的容錯(cuò)能力, 學(xué)會(huì)知錯(cuò)就改. 所以如果不是不可恢復(fù)性、導(dǎo)致系統(tǒng)無(wú)法正常工作的錯(cuò)誤, 如果發(fā)生了panic我們需要恢復(fù)程序, 讓程序繼續(xù)執(zhí)行,并且需要記錄到底犯了什么錯(cuò)誤
在Go語(yǔ)言中我們可以通過(guò)defer和recover來(lái)實(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) }
如果有異常寫(xiě)在defer中, 那么只有defer中的異常會(huì)被捕獲
package main import "fmt" func test2() { // 如果有異常寫(xiě)在defer中, 并且其它異常寫(xiě)在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) }
以上是“Go語(yǔ)言異常處理的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。