首頁 後端開發 Golang 在Go中什麼是defer?怎麼用?

在Go中什麼是defer?怎麼用?

Sep 01, 2021 pm 03:52 PM
golang

本文由go語言教學專欄給大家介紹,主題是關於go defer的學習使用,希望對需要的朋友有幫助!

什麼是defer?

在Go中,一個函數呼叫可以跟在一個defer關鍵字後面,形成一個延遲函數呼叫。
當一個函數呼叫被延遲後,它不會立即被執行。它將被推入由當前協程維護的一個延遲呼叫堆疊。當函數呼叫(可能是也可能不是延遲呼叫)返回並進入它的退出階段後,所有在此函數呼叫中已經被推入的延遲呼叫將被按照它們被推入堆疊的順序逆序執行。當所有這些延遲呼叫執行完畢後,此函數呼叫也就真正退出了。
舉個簡單的例子:

package mainimport "fmt"func sum(a, b int) {
    defer fmt.Println("sum函数即将返回")
    defer fmt.Println("sum函数finished")
    fmt.Printf("参数a=%v,参数b=%v,两数之和为%v\n", a, b, a+b)}func main() {
    sum(1, 2)}
登入後複製

output:

参数a=1,参数b=2,两数之和为3
sum函数finished
sum函数即将返回
登入後複製

事實上,每個協程都維護著兩個呼叫堆疊。

  • 一個是正常的函數呼叫堆疊。在此堆疊中,相鄰的兩個呼叫存在著呼叫關係。晚進入堆疊的呼叫被早進入堆疊的呼叫所呼叫。此堆疊中最早被推入的呼叫是對應協程的啟動呼叫。
  • 另一個堆疊是上面提到的延遲呼叫堆疊。處於延遲呼叫堆疊中的任意兩個呼叫之間不存在呼叫關係。

defer函數參數估值

  • #對於一個延遲函數調用,它的實參是在此調用被推入延遲呼叫堆疊的時候被估值的。
  • 一個匿名函數體內的表達式是在此函數被執行的時候才會被逐個估值的,不管此函數是被普通呼叫還是延遲呼叫。
    範例1:
package mainimport  "fmt"func  Print(a int) {fmt.Println("defer函数中a的值=", a)}func  main() {a := 10defer  Print(a)a = 1000fmt.Println("a的值=", a)}
登入後複製

output:

a的值= 1000
defer函数中a的值= 10
登入後複製

defer Print(a) 被加入到延遲呼叫堆疊的時候,a 的值是5,故defer Print(a ) 輸出的結果為5
範例2:

package mainimport "fmt"func main() {
    func() {
        for i := 0; i < 3; i++ {
            defer fmt.Println("a=", i)
        }
    }()

    fmt.Println()
    func() {
        for i := 0; i < 3; i++ {
            defer func() {
                fmt.Println("b=", i)
            }()
        }
    }()}
登入後複製

output:

a= 2
a= 1
a= 0

b= 3
b= 3
b= 3
登入後複製

第一個匿名函數循環中的i 是在fmt.Println函數呼叫被推入延遲呼叫堆疊的時候估的值,因此輸出結果是2,1,0 , 第二個匿名函數中的i 是匿名函數呼叫退出階段估的值(此時i 已經變成3了),故結果輸出:3, 3,3。
其實對第二個匿名函數呼叫略加修改,就能讓它輸出和匿名函數一相同的結果:

package mainimport "fmt"func main() {
    func() {
        for i := 0; i < 3; i++ {
            defer fmt.Println("a=", i)
        }
    }()

    fmt.Println()
    func() {
        for i := 0; i < 3; i++ {
            defer func(i int) {
                fmt.Println("b=", i)
            }(i)
        }
    }()}
登入後複製

output:

a= 2
a= 1
a= 0

b= 2
b= 1
b= 0
登入後複製

恐慌(panic)和恢復(defer recover)

Go不支援異常拋出和捕獲,而是建議使用返回值明確傳回錯誤。不過,Go支援一套和異常拋出/捕獲類似的機制。此機制稱為恐慌/恢復(panic/recover)機制。

我們可以呼叫內建函數panic來產生一個恐慌以使目前協程進入恐慌狀況。

進入恐慌狀況是另一種使當前函數呼叫開始返回的途徑。一旦函數呼叫產生一個恐慌,此函數呼叫將立即進入它的退出階段,在此函數呼叫中被推入堆疊的延遲呼叫將按照它們被推入的順序逆序執行。

透過在一個延遲函數呼叫之中呼叫內建函數recover,當前協程中的一個恐慌可以被消除,從而使得當前協程重新進入正常狀況。

在一個處於恐慌狀況的協程退出之前,其中的恐慌不會蔓延到其它協程。如果一個協程在恐慌狀況下退出,它將使整個程序崩潰。看下面的兩個例子:

package mainimport (
    "fmt"
    "time")func p(a, b int) int {
    return a / b}func main() {
    go func() {
        fmt.Println(p(1, 0))
    }()
    time.Sleep(time.Second)
    fmt.Println("程序正常退出~~~")}
登入後複製

output:

panic: runtime error: integer pide by zero

goroutine 6 [running]:
main.p(...)
        /Users/didi/Desktop/golang/defer.go:9
main.main.func1()
        /Users/didi/Desktop/golang/defer.go:14 +0x12
created by main.main
        /Users/didi/Desktop/golang/defer.go:13 +0x39exit status 2
登入後複製

p函數發生panic(除數為0),因為所在協程沒有恐慌恢復機制,導致整個程式崩潰。
如果p函數所在協程加上恐慌恢復(defer recover),程式便可正常退出。

package mainimport (
    "fmt"
    "time")func p(a, b int) int {
    return a / b}func main() {
    go func() {
        defer func() {
            v := recover()
            if v != nil {
                fmt.Println("恐慌被恢复了:", v)
            }
        }()
        fmt.Println(p(1, 0))
    }()
    time.Sleep(time.Second)
    fmt.Println("程序正常退出~~~")}
登入後複製

output:

恐慌被恢复了: runtime error: integer pide by zero
程序正常退出~~~
登入後複製

更多golang相關知識,請造訪golang#更多golang相關知識,請造訪

golang######教學欄!                                    #### ##

以上是在Go中什麼是defer?怎麼用?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

如何使用 Golang 安全地讀取和寫入檔案? 如何使用 Golang 安全地讀取和寫入檔案? Jun 06, 2024 pm 05:14 PM

在Go中安全地讀取和寫入檔案至關重要。指南包括:檢查檔案權限使用defer關閉檔案驗證檔案路徑使用上下文逾時遵循這些準則可確保資料的安全性和應用程式的健全性。

如何為 Golang 資料庫連線配置連線池? 如何為 Golang 資料庫連線配置連線池? Jun 06, 2024 am 11:21 AM

如何為Go資料庫連線配置連線池?使用database/sql包中的DB類型建立資料庫連線;設定MaxOpenConns以控制最大並發連線數;設定MaxIdleConns以設定最大空閒連線數;設定ConnMaxLifetime以控制連線的最大生命週期。

如何在 Golang 中將 JSON 資料保存到資料庫中? 如何在 Golang 中將 JSON 資料保存到資料庫中? Jun 06, 2024 am 11:24 AM

可以透過使用gjson函式庫或json.Unmarshal函數將JSON資料儲存到MySQL資料庫中。 gjson函式庫提供了方便的方法來解析JSON字段,而json.Unmarshal函數需要一個目標類型指標來解組JSON資料。這兩種方法都需要準備SQL語句和執行插入操作來將資料持久化到資料庫中。

Golang框架與Go框架:內部架構與外部特性對比 Golang框架與Go框架:內部架構與外部特性對比 Jun 06, 2024 pm 12:37 PM

GoLang框架與Go框架的差異體現在內部架構與外部特性。 GoLang框架基於Go標準函式庫,擴充其功能,而Go框架由獨立函式庫組成,以實現特定目的。 GoLang框架更靈活,Go框架更容易上手。 GoLang框架在效能上稍有優勢,Go框架的可擴充性更高。案例:gin-gonic(Go框架)用於建立RESTAPI,而Echo(GoLang框架)用於建立Web應用程式。

從前端轉型後端開發,學習Java還是Golang更有前景? 從前端轉型後端開發,學習Java還是Golang更有前景? Apr 02, 2025 am 09:12 AM

後端學習路徑:從前端轉型到後端的探索之旅作為一名從前端開發轉型的後端初學者,你已經有了nodejs的基礎,...

如何找出 Golang 正規表示式符合的第一個子字串? 如何找出 Golang 正規表示式符合的第一個子字串? Jun 06, 2024 am 10:51 AM

FindStringSubmatch函數可找出正規表示式匹配的第一個子字串:此函數傳回包含匹配子字串的切片,第一個元素為整個匹配字串,後續元素為各個子字串。程式碼範例:regexp.FindStringSubmatch(text,pattern)傳回符合子字串的切片。實戰案例:可用於匹配電子郵件地址中的域名,例如:email:="user@example.com",pattern:=@([^\s]+)$獲取域名match[1]。

golang框架開發實戰教學:常見疑問解答 golang框架開發實戰教學:常見疑問解答 Jun 06, 2024 am 11:02 AM

Go框架開發常見問題:框架選擇:取決於應用需求和開發者偏好,如Gin(API)、Echo(可擴展)、Beego(ORM)、Iris(效能)。安裝和使用:使用gomod指令安裝,導入框架並使用。資料庫互動:使用ORM庫,如gorm,建立資料庫連線和操作。身份驗證和授權:使用會話管理和身份驗證中間件,如gin-contrib/sessions。實戰案例:使用Gin框架建立一個簡單的部落格API,提供POST、GET等功能。

如何用 Golang 使用預先定義時區? 如何用 Golang 使用預先定義時區? Jun 06, 2024 pm 01:02 PM

Go語言中使用預先定義時區包含下列步驟:匯入"time"套件。透過LoadLocation函數載入特定時區。在建立Time物件、解析時間字串等操作中使用已載入的時區,進行日期和時間轉換。使用不同時區的日期進行比較,以說明預先定義時區功能的應用。

See all articles