前言
大家都知道golang的defer關鍵字,它可以在函數返回前執(zhí)行一些操作,最常用的就是打開一個資源(例如一個文件、數據庫連接等)時就用defer延遲關閉改資源,以免引起內存泄漏。本文主要給大家介紹了關于golang中defer的關鍵特性,分享出來供大家參考學習,下面話不多說,來一起看看詳細的介紹:
一、defer 的作用和執(zhí)行時機
go 的 defer 語句是用來延遲執(zhí)行函數的,而且延遲發(fā)生在調用函數 return 之后,比如
func a() int {
defer b()
return 0
}
b 的執(zhí)行是發(fā)生在 return 0 之后,注意 defer 的語法,關鍵字 defer 之后是函數的調用。
二、defer 的重要用途一:清理釋放資源
由于 defer 的延遲特性,defer 常用在函數調用結束之后清理相關的資源,比如
f, _ := os.Open(filename)
defer f.Close()
文件資源的釋放會在函數調用結束之后借助 defer 自動執(zhí)行,不需要時刻記住哪里的資源需要釋放,打開和釋放必須相對應。
用一個例子深刻詮釋一下 defer 帶來的便利和簡潔。
代碼的主要目的是打開一個文件,然后復制內容到另一個新的文件中,沒有 defer 時這樣寫:
func CopyFile(dstName, srcName string) (written int64, err error) {
src, err := os.Open(srcName)
if err != nil {
return
}
dst, err := os.Create(dstName)
if err != nil { //1
return
}
written, err = io.Copy(dst, src)
dst.Close()
src.Close()
return
}
代碼在 #1 處返回之后,src 文件沒有執(zhí)行關閉操作,可能會導致資源不能正確釋放,改用 defer 實現:
func CopyFile(dstName, srcName string) (written int64, err error) {
src, err := os.Open(srcName)
if err != nil {
return
}
defer src.Close()
dst, err := os.Create(dstName)
if err != nil {
return
}
defer dst.Close()
return io.Copy(dst, src)
}
src 和 dst 都能及時清理和釋放,無論 return 在什么地方執(zhí)行。
鑒于 defer 的這種作用,defer 常用來釋放數據庫連接,文件打開句柄等釋放資源的操作。
三、defer 的重要用途二:執(zhí)行 recover
被 defer 的函數在 return 之后執(zhí)行,這個時機點正好可以捕獲函數拋出的 panic,因而 defer 的另一個重要用途就是執(zhí)行 recover。
recover 只有在 defer 中使用才更有意義,如果在其他地方使用,由于 program 已經調用結束而提前返回而無法有效捕捉錯誤。
package main
import (
"fmt"
)
func main() {
defer func() {
if ok := recover(); ok != nil {
fmt.Println("recover")
}
}()
panic("error")
}
記住 defer 要放在 panic 執(zhí)行之前。
四、多個 defer 的執(zhí)行順序
defer 的作用就是把關鍵字之后的函數執(zhí)行壓入一個棧中延遲執(zhí)行,多個 defer 的執(zhí)行順序是后進先出 LIFO :
defer func() { fmt.Println("1") }()
defer func() { fmt.Println("2") }()
defer func() { fmt.Println("3") }()
輸出順序是 321。
這個特性可以對一個 array 實現逆序操作。
五、被 deferred 函數的參數在 defer 時確定
這是 defer 的特點,一個函數被 defer 時,它的參數在 defer 時進行計算確定,即使 defer 之后參數發(fā)生修改,對已經 defer 的函數沒有影響,什么意思?看例子:
func a() {
i := 0
defer fmt.Println(i)
i++
return
}
a 執(zhí)行輸出的是 0 而不是 1,因為 defer 時,i 的值是 0,此時被 defer 的函數參數已經進行執(zhí)行計算并確定了。
再看一個例子:
func calc(index string, a, b int) int {
ret := a + b
fmt.Println(index, a, b, ret)
return ret
}
func main() {
a := 1
b := 2
defer calc("1", a, calc("10", a, b))
a = 0
return
}
執(zhí)行代碼輸出
defer 函數的參數 第三個參數在 defer 時就已經計算完成并確定,第二個參數 a 也是如此,無論之后 a 變量是否修改都不影響。
六、被 defer 的函數可以讀取和修改帶名稱的返回值
func c() (i int) {
defer func() { i++ }()
return 1
}
被 defer 的函數是在 return 之后執(zhí)行,可以修改帶名稱的返回值,上面的函數 c 返回的是 2。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
參考資料
https://blog.golang.org/defer-panic-and-recover
您可能感興趣的文章:- 分析MongoDB和MySQL各自的關鍵特性、差別和優(yōu)勢
- golang1.16新特性速覽(推薦)
- go特性之數組與切片的問題
- django 中QuerySet特性功能詳解
- MongoDB的基本特性與內部構造的講解
- Django 2.0版本的新特性搶先看!
- Go語言區(qū)別于其他語言的特性