總結(jié)Go語言中defer的使用和注意要點(diǎn)
前言
defer是golang語言中的關(guān)鍵字,用于資源的釋放,會在函數(shù)返回之前進(jìn)行調(diào)用。
一般采用如下模式:
f,err := os.Open(filename)
if err != nil {
panic(err)
}
defer f.Close()
如果有多個defer表達(dá)式,調(diào)用順序類似于棧,越后面的defer表達(dá)式越先被調(diào)用。
延時調(diào)用函數(shù)的語法如下:
defer func_name(param-list)
當(dāng)一個函數(shù)調(diào)用前有關(guān)鍵字 defer 時, 那么這個函數(shù)的執(zhí)行會推遲到包含這個 defer 語句的函數(shù)即將返回前才執(zhí)行. 例如:
func main() {
defer fmt.Println("Fourth")
fmt.Println("First")
fmt.Println("Third")
}
最后打印順序如下:
First Second Third
需要注意的是, defer 調(diào)用的函數(shù)參數(shù)的值 defer 被定義時就確定了.
例如:
i := 1
defer fmt.Println("Deferred print:", i)
i++
fmt.Println("Normal print:", i)
打印的內(nèi)容如下:
Normal print: 2 Deferred print: 1
因此我們知道, 在 "defer fmt.Println("Deferred print:", i)" 調(diào)用時, i 的值已經(jīng)確定了, 因此相當(dāng)于 defer fmt.Println("Deferred print:", 1) 了.
需要強(qiáng)調(diào)的時, defer 調(diào)用的函數(shù)參數(shù)的值在 defer 定義時就確定了, 而 defer 函數(shù)內(nèi)部所使用的變量的值需要在這個函數(shù)運(yùn)行時才確定.
例如:
func f1() (r int) {
r = 1
defer func() {
r++
fmt.Println(r)
}()
r = 2
return
}
func main() {
f1()
}
上面的例子中, 最終打印的內(nèi)容是 "3", 這是因?yàn)樵?"r = 2" 賦值之后, 執(zhí)行了 defer 函數(shù), 因此在這個函數(shù)內(nèi), r 的值是2了, 自增后變?yōu)?.
defer 順序
如果有多個defer 調(diào)用, 則調(diào)用的順序是先進(jìn)后出的順序, 類似于入棧出棧一樣:
func main() {
defer fmt.Println(1)
defer fmt.Println(2)
defer fmt.Println(3)
defer fmt.Println(4)
}
最先執(zhí)行的是 "fmt.Println(4)" , 接著是 "fmt.Println(3)" 依次類推, 最后的輸出如下:
4 3 2 1
defer 注意要點(diǎn)
defer 函數(shù)調(diào)用的執(zhí)行時機(jī)是外層函數(shù)設(shè)置返回值之后, 并且在即將返回之前.
例如:
func f1() (r int) {
defer func() {
r++
}()
return 0
}
func main() {
fmt.Println(f1())
}
上面 fmt.Println(f1()) 打印的是什么呢? 很多朋友可能會認(rèn)為打印的是0, 但是正確答案是 1. 這是為什么呢?
要弄明白這個問題, 我們需要牢記兩點(diǎn)
1、defer 函數(shù)調(diào)用的執(zhí)行時機(jī)是外層函數(shù)設(shè)置返回值之后, 并且在即將返回之前
2、return XXX 操作并不是原子的.
我們將上面的例子改寫一下大家就很明白了:
func f1() (r int) {
defer func() {
r++
}()
r = 0
return
}
當(dāng)進(jìn)行賦值操作 "r = 0" 后, 才調(diào)用 defer 函數(shù), 最后才是返回語句.
因此上面的代碼等效于:
func f1() (r int) {
r = 0
func() {
r++
}()
return
}
接下來我們再來看一個更有意思的例子:
func double(x int) int {
return x + x
}
func triple(x int) (r int) {
defer func() {
r += x
}()
return double(x)
}
func main() {
fmt.Println(triple(3))
}
如果我們已經(jīng)理解了上面所說的內(nèi)容的話, 那么 triple 函數(shù)就很好理解了, 它實(shí)際上是:
func triple(x int) (r int) {
r = double(x)
func() {
r += x
}()
return
}
defer 表達(dá)式的使用場景
defer 通常用于 open/close, connect/disconnect, lock/unlock 等這些成對的操作, 來保證在任何情況下資源都被正確釋放. 在這個角度來說, defer 操作和 Java 中的 try ... finally 語句塊有異曲同工之處.
例如:
var mutex sync.Mutex
var count = 0
func increment() {
mutex.Lock()
defer mutex.Unlock()
count++
}
在increment 函數(shù)中, 我們?yōu)榱吮苊飧倯B(tài)條件的出現(xiàn), 而使用了 Mutex 進(jìn)行加鎖. 而在進(jìn)行并發(fā)編程時, 加鎖了卻忘記(或某種情況下 unlock 沒有被執(zhí)行), 往往會造成災(zāi)難性的后果. 為了在任意情況下, 都要保證在加鎖操作后, 都進(jìn)行對應(yīng)的解鎖操作, 我們可以使用 defer 調(diào)用解鎖操作.
總結(jié)
以上就是這篇文章的全部內(nèi)容,希望對大家的學(xué)習(xí)或者工作帶來一定的幫助。如果有疑問大家可以留言交流。
相關(guān)文章
Go語言切片前或中間插入項(xiàng)與內(nèi)置copy()函數(shù)詳解
這篇文章主要介紹了Go語言切片前或中間插入項(xiàng)與內(nèi)置copy()函數(shù)詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-04-04
go語言轉(zhuǎn)換json字符串為json數(shù)據(jù)的實(shí)現(xiàn)
本文主要介紹了go語言轉(zhuǎn)換json字符串為json數(shù)據(jù)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-03-03
詳解如何在Golang中實(shí)現(xiàn)CORS(跨域)
很多時候,需要允許Web應(yīng)用程序在不同域之間(跨域)實(shí)現(xiàn)共享資源,本文將簡介跨域、CORS的概念,以及如何在Golang中如何實(shí)現(xiàn)CORS,文中有詳細(xì)的示例代碼,需要的朋友可以參考下2023-10-10
Golang算法問題之整數(shù)拆分實(shí)現(xiàn)方法分析
這篇文章主要介紹了Golang算法問題之整數(shù)拆分實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了Go語言數(shù)值運(yùn)算與數(shù)組遍歷相關(guān)操作技巧,需要的朋友可以參考下2017-02-02
Go語言學(xué)習(xí)之new函數(shù)的用法詳解
這篇文章主要為大家詳細(xì)介紹了Go語言中new()函數(shù)的相關(guān)知識以及具體用法,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價值,感興趣的小伙伴可以了解一下2023-05-05

