Go語(yǔ)言異常處理案例解析
更新時(shí)間:2021年07月29日 08:46:00 作者:極客江南
這篇文章主要介紹了Go語(yǔ)言異常處理案例解析,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
異常處理
- 程序運(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è)名稱叫做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...))
}
- 應(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
恢復(fù)程序
- 程序和人一樣都需要具備一定的容錯(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)
}
到此這篇關(guān)于Go語(yǔ)言異常處理案例解析的文章就介紹到這了,更多相關(guān)Go語(yǔ)言異常處理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang中重復(fù)錯(cuò)誤處理的優(yōu)化方法
這篇文章主要給大家介紹了關(guān)于Golang中重復(fù)錯(cuò)誤處理優(yōu)化的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Golang具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
Go-ethereum?解析ethersjs中產(chǎn)生的簽名信息思路詳解
這篇文章主要介紹了Go-ethereum?解析ethersjs中產(chǎn)生的簽名信息,我們解析簽名的需要知道,簽名的消息,簽名,和公鑰,按照這個(gè)思路,我們可以通過(guò)ethers實(shí)現(xiàn)消息的簽名,也可以通過(guò)go-ethereum實(shí)現(xiàn),需要的朋友可以參考下2022-08-08
Goland 斷點(diǎn)調(diào)試Debug的操作
這篇文章主要介紹了Goland 斷點(diǎn)調(diào)試Debug的操作方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-04-04
Golang使用Apache PLC4X連接modbus的示例代碼
Modbus是一種串行通信協(xié)議,是Modicon公司于1979年為使用可編程邏輯控制器(PLC)通信而發(fā)表,這篇文章主要介紹了Golang使用Apache PLC4X連接modbus的示例代碼,需要的朋友可以參考下2024-07-07
Go設(shè)計(jì)模式之訪問(wèn)者模式講解和代碼示例
訪問(wèn)者是一種行為設(shè)計(jì)模式, 允許你在不修改已有代碼的情況下向已有類層次結(jié)構(gòu)中增加新的行為,本文將通過(guò)代碼示例給大家詳細(xì)的介紹一下Go設(shè)計(jì)模式之訪問(wèn)者模式,需要的朋友可以參考下2023-08-08

