欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入探究Go語(yǔ)言的錯(cuò)誤策略與異常機(jī)制

 更新時(shí)間:2024年02月03日 09:37:39   作者:鼠鼠我捏,要死了捏  
本文深入探討了Go語(yǔ)言的錯(cuò)誤策略與異常機(jī)制,主要介紹了錯(cuò)誤處理的重要性,以及Go語(yǔ)言中的錯(cuò)誤類型和處理函數(shù),此外還討論了Go語(yǔ)言的異常機(jī)制,包括panic和recover函數(shù)的使用,需要的朋友可以參考下

前言

作為開發(fā)者來(lái)說(shuō),我們沒(méi)辦法保證程序在運(yùn)行過(guò)程中永遠(yuǎn)不會(huì)出現(xiàn)異常,對(duì)于異常,在很多編程語(yǔ)言中,可以用 try-catch語(yǔ)句來(lái)捕獲,而Go語(yǔ)言的開發(fā)者顯然覺(jué)得 try-catch被濫用了,因此 Go不支持使用 try-catch語(yǔ)句捕獲異常處理。

那么,Go語(yǔ)言是如何定義和處理程序的異常呢?

Go語(yǔ)言的將程序運(yùn)行出現(xiàn)的問(wèn)題分為兩種:錯(cuò)誤(error)和異常(exception),錯(cuò)誤是程序(比如一個(gè)函數(shù))運(yùn)行的預(yù)期結(jié)果之一,當(dāng)你調(diào)用一個(gè)函數(shù)時(shí),就應(yīng)該處理出現(xiàn)錯(cuò)誤的情況,而異常是不可預(yù)期的(當(dāng)然也可以手動(dòng)觸發(fā))且會(huì)導(dǎo)致程序中斷運(yùn)行的嚴(yán)重錯(cuò)誤。

Go錯(cuò)誤處理策略

編寫Go語(yǔ)言程序,一般推薦通過(guò)函數(shù)的最后一個(gè)返回值告訴調(diào)用者函數(shù)是否執(zhí)行成功,可以分兩種情況來(lái)討論:

第一種情況,如果函數(shù)的執(zhí)行結(jié)果只有正確與失敗,那么返回值可以是 boolean類型:

func exists(key string) bool {
	//to check if the key is exists
	return true
}
 
if ok := exists("test"); ok {
	// do something
}

第二種情況,如果要讓調(diào)用者得到更詳細(xì)的錯(cuò)誤信息,顯然只返回一個(gè)布爾值是不夠的,這時(shí)候可以返回一個(gè) error類型,error類型是一個(gè)接口,只有一個(gè) Error方法的接口:

type error interface {
	Error() string
}

通過(guò) error類型的 Error方法,可以獲得更詳細(xì)的錯(cuò)誤信息,方便調(diào)用者做進(jìn)一步的處理,因此在 Go標(biāo)準(zhǔn)庫(kù)的方法和函數(shù)中,一般都會(huì)將 error類型作為最后一個(gè)返回值,比如我們以前文章中講到的 os包下的 Open和 Create函數(shù):

package os
 
func Open(name string) (*File, error) {
	return OpenFile(name, O_RDONLY, 0)
}
 
func Create(name string) (*File, error) {
	return OpenFile(name, O_RDWR|O_CREATE|O_TRUNC, 0666)
}

當(dāng)函數(shù)最后一個(gè)返回值 error不為 nil時(shí),表示執(zhí)行不成功,此時(shí)其他的返回值就應(yīng)該忽略:

file,err := os.Open("./my.dat")
 
if err != nil{
  	return err
}
 
defer file.Close()

上面代碼中,如果 error不為 nil,那么變量 file則為 nil,表示無(wú)法獲得一個(gè)文件句柄,這時(shí)候直接返回錯(cuò)誤,因?yàn)槿绻倮^續(xù)往下執(zhí)行,可能會(huì)引發(fā)程序崩潰。

當(dāng)然并不是所有情況下一遇到返回的 error不為 nil,就要拋棄其他返回值,比如使用 Read()方法讀取文件時(shí):

Read(p []byte) (n int, err error)

在讀取文件到結(jié)尾時(shí) Read方法會(huì)返回一個(gè) io.EOF的錯(cuò)誤:

var EOF = errors.New("EOF")

此時(shí)雖然 error不為 nil,但仍然應(yīng)該把讀取到的字節(jié)數(shù)組保存,而不是拋棄掉。

創(chuàng)建error的幾種方式

當(dāng)我們開發(fā)自己的函數(shù)時(shí),也可以創(chuàng)建自己的錯(cuò)誤類型,有以下幾種方式:

errors包

errors包下的 New()函數(shù)可以創(chuàng)建一個(gè)只有文本信息的 error類型:

package main
 
func main() {
	var s = []int{1, 2, 3}
	s[3] = 10
}

fmt包

fmt包下的 Errorf函數(shù)可以將文本格式后作為error類型的錯(cuò)誤信息,并返回一個(gè)error類型,因此其作用與errors.New函數(shù)類似

func getFile(name string)(*os.file,error){
	if name == ""{
		return nil,fmt.Errorf("file name could not be empty")
	}
}

fmt.Errorf函數(shù)還能封裝其他 error類型,再返回一個(gè)新的 error類型,以此形成一條完整的錯(cuò)誤鏈條:

doc, err := html.Parse(resp.Body)
resp.Body.Close()
if err != nil {
	return nil, fmt.Errorf("parsing %s as HTML: %v", url,err)
}

自定義錯(cuò)誤類型

其實(shí),上面兩種創(chuàng)建錯(cuò)誤類型的方式,本質(zhì)上都是實(shí)現(xiàn) error接口,我們也可以創(chuàng)建一個(gè)擁有 Error方法的類型來(lái)實(shí)現(xiàn) error接口:

type Result struct {
	Code    int
	Message string
	Data    interface{}
}
 
func (r Result) Error() string {
	return r.Message
}

如何處理錯(cuò)誤

當(dāng)調(diào)用的函數(shù)或方法的返回值有 error類型時(shí),最簡(jiǎn)單的當(dāng)然可以選擇直接忽略錯(cuò)誤,不過(guò)更恰當(dāng)?shù)姆绞绞翘幚韺?duì)應(yīng)的錯(cuò)誤,有以下幾種處理策略:

直接返回錯(cuò)誤

對(duì)于函數(shù)來(lái)說(shuō),如果在執(zhí)行時(shí)遇到錯(cuò)誤,可以直接返回給上層調(diào)用者:

func SendMessage(url string) error {
	if url == ""{
		return errors.New("url can't not be empty")
	}
	_, err := http.Get(url)
	if err != nil {
		return err
	}
	return nil
}

記錄日志并繼續(xù)運(yùn)行

當(dāng)調(diào)用函數(shù)時(shí)遇到返回的錯(cuò)誤,如果不影響程序運(yùn)行,也可以選擇記錄錯(cuò)誤日志并往下執(zhí)行:

if err := SendMessage("https://xxx/sendMessage");err != nil{
	log.Printf("the message sending been broken by : %v\n", err)
}
 

記錄日志并結(jié)束運(yùn)行

如果錯(cuò)誤影響程序的執(zhí)行,也可以記錄日志后,退出程序執(zhí)行:

if err := SendMessage("https://xxx/sendMessage");err != nil{
	log.Printf("the message sending been broken by : %v\n", err)
	os.Exit(1)
}

記錄日志并退出執(zhí)行,直接用 log包的 Fatalf函數(shù)就可以做到,因此上面代碼的更簡(jiǎn)潔做法是:

if err := SendMessage("https://xxx/sendMessage");err != nil{
	log.Fatalf("the message sending been broken by : %v\n", err)
}

Go異常處理機(jī)制

在Go語(yǔ)言中,異常是指會(huì)引發(fā)程序崩潰無(wú)法繼續(xù)運(yùn)行的錯(cuò)誤,比如數(shù)組越界或者空指針引用等情況,這時(shí)候會(huì)觸發(fā) panic異常:

package main
 
func main() {
	var s = []int{1, 2, 3}
	s[3] = 10
}

上面程序運(yùn)行結(jié)果如下,可以看出觸發(fā)了數(shù)組越界的異常:

panic: runtime error: index out of range [3] with length 3

無(wú)論是在主協(xié)程還是子協(xié)程中,一旦觸發(fā) panic異常,整個(gè)程序都會(huì)終止運(yùn)行。

panic函數(shù)

除了數(shù)組越界等不可預(yù)測(cè)的異常會(huì)自動(dòng)觸發(fā) panic,也可以手動(dòng)調(diào)用 panic函數(shù)觸發(fā)異常來(lái)終止程序的運(yùn)行:

func loadConfig(path string){
	panic("can't load the config file on path " + path)
}

一個(gè)良好的程序最好不要主動(dòng)調(diào)用 panic函數(shù),尤其是開發(fā)類庫(kù)的時(shí)候,最好通過(guò)返回 error類型來(lái)告訴調(diào)用者發(fā)生了什么錯(cuò)誤,而不是觸發(fā) panic導(dǎo)致程序終止運(yùn)行。

recover函數(shù)

當(dāng)發(fā)生 panic異常時(shí),如果不捕獲得異常,那么程序就是終止運(yùn)行,在Go語(yǔ)言中,可以用 defer語(yǔ)句和 recover函數(shù)的模式來(lái)捕獲 panic異常:

package main
 
import (
	"fmt"
	"log"
)
 
func main() {
 
	n1 := FindElementByIndex(1)
	fmt.Println(n1)
  
	n2 := FindElementByIndex(4)
	fmt.Println(n2)
}
 
func FindElementByIndex(index int) int {
	defer func() {
		if e := recover(); e != nil {
			log.Fatal(e)
		}
	}()
	s := []int{1, 2, 3, 4}
	return s[index]
}
 

總結(jié)

本文深入探討了Go語(yǔ)言的錯(cuò)誤策略與異常機(jī)制。主要介紹了錯(cuò)誤處理的重要性,以及Go語(yǔ)言中的錯(cuò)誤類型和處理函數(shù)。此外還討論了Go語(yǔ)言的異常機(jī)制,包括panic和recover函數(shù)的使用。通過(guò)合理的錯(cuò)誤處理和異常處理,我們可以提高代碼的可維護(hù)性和可靠性,減少潛在的bug和故障。希望本文對(duì)您有幫助,感謝閱讀~

以上就是深入探究Go語(yǔ)言的錯(cuò)誤策略與異常機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于Go錯(cuò)誤策略與異常機(jī)制的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • jenkins構(gòu)建go及java項(xiàng)目的方法

    jenkins構(gòu)建go及java項(xiàng)目的方法

    這篇文章主要介紹了jenkins構(gòu)建go及java項(xiàng)目,本文通過(guò)圖文實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值了,需要的朋友可以參考下
    2021-04-04
  • Go語(yǔ)言并發(fā)編程基礎(chǔ)上下文概念詳解

    Go語(yǔ)言并發(fā)編程基礎(chǔ)上下文概念詳解

    這篇文章主要為大家介紹了Go語(yǔ)言并發(fā)編程基礎(chǔ)上下文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Go?interface{}?轉(zhuǎn)切片類型的實(shí)現(xiàn)方法

    Go?interface{}?轉(zhuǎn)切片類型的實(shí)現(xiàn)方法

    本文主要介紹了Go?interface{}?轉(zhuǎn)切片類型的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • Go語(yǔ)言實(shí)現(xiàn)聊天小工具的示例代碼

    Go語(yǔ)言實(shí)現(xiàn)聊天小工具的示例代碼

    這篇文章主要為大家詳細(xì)介紹了如何利用Go語(yǔ)言實(shí)現(xiàn)聊天小工具,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Go語(yǔ)言Gin框架獲取請(qǐng)求參數(shù)的兩種方式

    Go語(yǔ)言Gin框架獲取請(qǐng)求參數(shù)的兩種方式

    在添加路由處理函數(shù)之后,就可以在路由處理函數(shù)中編寫業(yè)務(wù)處理代碼了,而編寫業(yè)務(wù)代碼第一件事一般就是獲取HTTP請(qǐng)求的參數(shù)吧,Gin框架在net/http包的基礎(chǔ)上封裝了獲取參數(shù)的方式,本文小編給大家介紹了獲取參數(shù)的兩種方式,需要的朋友可以參考下
    2024-01-01
  • Go語(yǔ)言程序查看和診斷工具詳解

    Go語(yǔ)言程序查看和診斷工具詳解

    這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言程序查看和診斷工具,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • Go?分布式鏈路追蹤實(shí)現(xiàn)原理解析

    Go?分布式鏈路追蹤實(shí)現(xiàn)原理解析

    分布式鏈路追蹤作為解決分布式應(yīng)用可觀測(cè)問(wèn)題的重要技術(shù),愈發(fā)成為分布式應(yīng)用不可缺少的基礎(chǔ)設(shè)施,本文將詳細(xì)介紹分布式鏈路的核心概念、架構(gòu)原理和相關(guān)開源標(biāo)準(zhǔn)協(xié)議,并分享我們?cè)趯?shí)現(xiàn)無(wú)侵入 Go 采集 Sdk 方面的一些實(shí)踐,需要的朋友可以參考下
    2022-06-06
  • Go語(yǔ)言實(shí)現(xiàn)二進(jìn)制與十進(jìn)制互轉(zhuǎn)的示例代碼

    Go語(yǔ)言實(shí)現(xiàn)二進(jìn)制與十進(jìn)制互轉(zhuǎn)的示例代碼

    這篇文章主要和大家詳細(xì)介紹了Go語(yǔ)言中實(shí)現(xiàn)二進(jìn)制與十進(jìn)制互相轉(zhuǎn)換的示例代碼,文中的代碼簡(jiǎn)潔易懂,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-05-05
  • golang xorm及time.Time自定義解決json日期格式的問(wèn)題

    golang xorm及time.Time自定義解決json日期格式的問(wèn)題

    這篇文章主要介紹了golang xorm及time.Time自定義解決json日期格式的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • 詳解Go如何優(yōu)雅的對(duì)時(shí)間進(jìn)行格式化

    詳解Go如何優(yōu)雅的對(duì)時(shí)間進(jìn)行格式化

    這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言中是如何優(yōu)雅的對(duì)時(shí)間進(jìn)行格式化的,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起了解一下
    2023-06-06

最新評(píng)論