go語言defer語句的用法
defer後面必須是函式呼叫語句,不能是其他語句,否則編譯器會出錯。
這個例子中defer後面使用的是n++指令,不是乙個函式呼叫語句,編譯器就報錯:package main
import "log"
func foo(n int) int
func main()
defer後面的函式在defer語句所在的函式執行結束的時候會被呼叫;我們檢視一下彙編嗎,看看defer是在什麼時候被執行的:# command-line-arguments
./main.go:6: expression in defer must be function call
./main.go:6: syntax error: unexpected ++ at end of statement
定義兩個函式foo1和foo2,功能和**都是一樣,只是其中乙個包含defer語句,另乙個沒有。
這是foo1的彙編**:func foo1(i int) int
func foo2(i int) int
再看foo2的彙編**:func foo1(i int) int
通過比較很容易看出foo2有兩處需要注意,第一處是defer foo()語句的翻譯,這個翻譯我沒有細看懂,我猜是準備foo的函式引數(如果有),然後儲存這些引數值和foo的位址,註冊到系統(runtime.deferproc);另一處是return指令的翻譯,return指令的執行分三步,第一步拷貝return值到返回值記憶體位址,第二步會呼叫runtime.deferreturn去執行前面註冊的defer函式,第三部再執行ret彙編指令。func foo2(i int) int
有兩個常見的defer語句應用場景是:
在開啟輸入檔案輸出檔案後,不管後面的**流程如何影響,這兩個檔案能夠被自動關閉。func copyfile(dstname, srcname string) (written int64, err error)
defer src.close()
dst, err := os.create(dstname)
if err != nil
defer dst.close()
// other codes
return io.copy(dst, src)
}
確保mu鎖能夠在函式foo退出之後自動釋放。func foo(...)
我們注意到defer函式的執行是在defer指令所在函式的執行結束之後,那麼如何才能在所在函式的中間就釋放呢,比如前面例子,在foo入口鎖住了lock,而如果foo後半段的**執行時間比較長,而此時又不需要繼續保持住鎖,該怎麼辦呢?
我們希望能在time-consuming operation 之前就釋放鎖,而不是等到整個foo返回。這有兩個辦法,乙個是根據邏輯,把foo拆分兩部分,前半部分需要鎖,後半部分不需要鎖;另乙個辦法是使用匿名函式:func foo()
// time-consuming operating with object
...}
執行結果:package main
import "log"
import "time"
import "sync"
var mu sync.mutex
func lock()
func unlock()
func foo() int ()
time.sleep(1 * time.second)
log.printf("return")
return 0;
}func main()
從日誌我們可以看出mu鎖在sleep語句之前已經被釋放了,而不是需要等到foo函式結束的時候才釋放。$ ./main
2017/09/30 22:18:58 lock
2017/09/30 22:18:58 inner
2017/09/30 22:18:58 unlock
2017/09/30 22:18:59 return
2017/09/30 22:18:59 r= 0
如果函式裡面有多條defer指令,他們的執行順序是反序,即後定義的defer先執行。
執行結果如下,可以看出他們的呼叫順序:package main
import "log"
import "time"
func foo(n int) int
func main()
defer函式的引數是在defer語句出現的位置做計算的,而不是在函式執行的時候做計算的,即所在函式結束的時候計算的。2017/09/30 19:22:03 3333
2017/09/30 19:22:03 2222
2017/09/30 19:22:03 1111
其執行結果是:package main
import "log"
func foo(n int) int
func main()
可以看到defer函式的位置時n的值為100,儘管在函式foo結束的時候n的值已經是200了,但是defer語句本身所處的位置時刻,即foo函式入口時n為100,所以最終defer函式列印出來的n值為100。2017/09/30 19:25:10 n1= 100
2017/09/30 19:25:10 n2= 200
2017/09/30 19:25:10 n= 100
前面我們提到defer後面只能是一條函式呼叫指令;而實際情況下經常會需要邏輯執行,會有分支,條件,而不是簡單的乙個log.print指令;那怎麼處理這種情況呢,我們可以把這些邏輯指令一起定義成乙個函式,然後再呼叫這些函式就行了,命名函式或者匿名函式都可以,下面是乙個匿名函式的例子:
執行結果:package main
import "log"
import _ "time"
func foo(n int) int ()
n += 100
log.println("n2=", n)
return n
}func main()
眼尖的同學會發現其中的問題;為什麼n列印出來是300呢,不是明明說好defer函式的引數值在它出現時候計算,而不是在執行的時候計算的嗎,n應該列印出200才對啊?2017/09/30 19:30:58 n1= 100
2017/09/30 19:30:58 n2= 200
2017/09/30 19:30:58 n= 300
同學,仔細看一下原文:defer函式的引數在defer語句出現的位置計算,不是在defer函式執行的時刻計算;人家明明說的很清楚,defer函式的引數,請問這裡n是引數嗎,不是哎,這裡引用的是宿主函式的區域性變數,而不是引數;所以它拿到的是執行時刻的值。
這就引發出下乙個注意事項。
執行結果為:package main
import "log"
func foo1(i *int) int ()
log.printf("i=%d", *i)
return *i
}func foo2(i *int) (r int) ()
log.printf("i=%d", *i)
return *i
}func main()
這個例子其實有一點拗口的。$ go build main.go && ./main
2017/09/30 20:01:00 i=100
2017/09/30 20:01:00 i=300, r=100
2017/09/30 20:01:00 i=100
2017/09/30 20:01:00 i=100, r=300
foo1 return指令前(i==100, ret==0),return指令後(i==100, ret=100),然後呼叫defer函式後(i==300,r==100),defer函式增加了i;main函式收到(i==300, r==100)
foo2 return指令前(i==100, ret==0),return指令後(i==100, ret=100),然後呼叫defer函式後(i==100,r==300),defer函式增加了ret;main函式收到(i==100, r==300)
go語言 defer 高階
go語言defer語句的用法 defer後面必須是函式呼叫語句,不能是其他語句,否則編譯器會出錯。package main import log func foo n int int func main 這個例子中defer後面使用的是n 指令,不是乙個函式呼叫語句,編譯器就報錯 command l...
Go語言defer學習小結
延時呼叫函式語法 defer func name param list defer func 關鍵字defer修飾的函式,值得關注的有以下幾點 1 函式會被推遲到包含這個defer語句的函式即將返回前才被呼叫執行。這點需要理解defer的工作機制,大致為defer出現的地方,插入指令call run...
go語言defer的使用
go語言的defer 延遲執行 語句,是在函式結束前執行,而如果在函式中有多個defer語句時,會像乙個堆疊一樣,先進後出,後進先出。defer語句在進行一些開啟資源的操作時,遇到錯誤需要提前返回,在返回前你需要關閉相應的資源,不然很容易造成資源洩露等問題上很試用。舉個defer語句簡單的使用如下 ...