溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

Go語(yǔ)言異常處理的示例分析

發(fā)布時(shí)間:2021-07-29 10:10:47 來(lái)源:億速云 閱讀:119 作者:小新 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要為大家展示了“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

恢復(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)
}

以上是“Go語(yǔ)言異常處理的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(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)容。

AI