亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

深入探究Go語言的錯誤策略與異常機制

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

前言

作為開發(fā)者來說,我們沒辦法保證程序在運行過程中永遠不會出現異常,對于異常,在很多編程語言中,可以用 try-catch語句來捕獲,而Go語言的開發(fā)者顯然覺得 try-catch被濫用了,因此 Go不支持使用 try-catch語句捕獲異常處理。

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

Go語言的將程序運行出現的問題分為兩種:錯誤(error)和異常(exception),錯誤是程序(比如一個函數)運行的預期結果之一,當你調用一個函數時,就應該處理出現錯誤的情況,而異常是不可預期的(當然也可以手動觸發(fā))且會導致程序中斷運行的嚴重錯誤。

Go錯誤處理策略

編寫Go語言程序,一般推薦通過函數的最后一個返回值告訴調用者函數是否執(zhí)行成功,可以分兩種情況來討論:

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

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

第二種情況,如果要讓調用者得到更詳細的錯誤信息,顯然只返回一個布爾值是不夠的,這時候可以返回一個 error類型,error類型是一個接口,只有一個 Error方法的接口:

type error interface {
	Error() string
}

通過 error類型的 Error方法,可以獲得更詳細的錯誤信息,方便調用者做進一步的處理,因此在 Go標準庫的方法和函數中,一般都會將 error類型作為最后一個返回值,比如我們以前文章中講到的 os包下的 Open和 Create函數:

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)
}

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

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

上面代碼中,如果 error不為 nil,那么變量 file則為 nil,表示無法獲得一個文件句柄,這時候直接返回錯誤,因為如果再繼續(xù)往下執(zhí)行,可能會引發(fā)程序崩潰。

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

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

在讀取文件到結尾時 Read方法會返回一個 io.EOF的錯誤:

var EOF = errors.New("EOF")

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

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

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

errors包

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

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

fmt包

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

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

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

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

自定義錯誤類型

其實,上面兩種創(chuàng)建錯誤類型的方式,本質上都是實現 error接口,我們也可以創(chuàng)建一個擁有 Error方法的類型來實現 error接口:

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

如何處理錯誤

當調用的函數或方法的返回值有 error類型時,最簡單的當然可以選擇直接忽略錯誤,不過更恰當的方式是處理對應的錯誤,有以下幾種處理策略:

直接返回錯誤

對于函數來說,如果在執(zhí)行時遇到錯誤,可以直接返回給上層調用者:

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ù)運行

當調用函數時遇到返回的錯誤,如果不影響程序運行,也可以選擇記錄錯誤日志并往下執(zhí)行:

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

記錄日志并結束運行

如果錯誤影響程序的執(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函數就可以做到,因此上面代碼的更簡潔做法是:

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

Go異常處理機制

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

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

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

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

無論是在主協(xié)程還是子協(xié)程中,一旦觸發(fā) panic異常,整個程序都會終止運行。

panic函數

除了數組越界等不可預測的異常會自動觸發(fā) panic,也可以手動調用 panic函數觸發(fā)異常來終止程序的運行:

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

一個良好的程序最好不要主動調用 panic函數,尤其是開發(fā)類庫的時候,最好通過返回 error類型來告訴調用者發(fā)生了什么錯誤,而不是觸發(fā) panic導致程序終止運行。

recover函數

當發(fā)生 panic異常時,如果不捕獲得異常,那么程序就是終止運行,在Go語言中,可以用 defer語句和 recover函數的模式來捕獲 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]
}
 

總結

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

以上就是深入探究Go語言的錯誤策略與異常機制的詳細內容,更多關于Go錯誤策略與異常機制的資料請關注腳本之家其它相關文章!

相關文章

  • golang內存對齊的項目實踐

    golang內存對齊的項目實踐

    本文主要介紹了golang內存對齊的項目實踐,內存對齊不僅有助于提高內存訪問效率,還確保了與硬件接口的兼容性,是Go語言編程中不可忽視的重要優(yōu)化手段,下面就來介紹一下
    2025-02-02
  • go語言中基本數據類型及應用快速了解

    go語言中基本數據類型及應用快速了解

    這篇文章主要為大家介紹了go語言中基本數據類型應用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-07-07
  • Go中跨域Cors中間件的實現

    Go中跨域Cors中間件的實現

    本文主要介紹了Go中跨域Cors中間件的實現,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-06-06
  • golang切片拷貝的實現

    golang切片拷貝的實現

    在Golang中,切片的淺拷貝只復制指向對象的指針,而深拷貝則復制數據本身,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-10-10
  • go mock模擬接口的實現

    go mock模擬接口的實現

    本文主要介紹了go mock模擬接口的實現,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-06-06
  • Go語言使用kafka-go實現Kafka消費消息

    Go語言使用kafka-go實現Kafka消費消息

    本篇文章主要介紹了使用kafka-go庫消費Kafka消息,包含FetchMessage和ReadMessage的區(qū)別和適用場景,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的可以了解一下
    2024-12-12
  • Go語言版本管理module以及go.sum詳解

    Go語言版本管理module以及go.sum詳解

    本文介紹了Go語言版本管理go module以及go.sum詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-08-08
  • golang?使用chromedp獲取頁面請求日志network

    golang?使用chromedp獲取頁面請求日志network

    這篇文章主要為大家介紹了golang?使用chromedp獲取頁面請求日志network方法實例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-11-11
  • Go語言中三種容器類型的數據結構詳解

    Go語言中三種容器類型的數據結構詳解

    在?Go?語言中,有三種主要的容器類型用于存儲和操作集合數據這篇文章主要為大家介紹了三者的使用與區(qū)別,感興趣的小伙伴可以跟隨小編一起學習一下
    2025-02-02
  • go slice 數組和切片使用區(qū)別示例解析

    go slice 數組和切片使用區(qū)別示例解析

    這篇文章主要為大家介紹了go slice 數組和切片使用區(qū)別示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01

最新評論