這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
return和defer的順序:
return先返回,再運行defer,可以修改傳回值:
package mainimport ( "fmt")func main() { fmt.Println(test())}//傳回值為2func test() (result int) { defer func() { result++ }() return 1 //return函數先返回 result被賦值為1 然後再result++}
這裡有個比較好玩兒的點,因為傳回值是具名的,所以我們return 1是對result賦值為1了。
另外如果是不具名的傳回值,但是你return了一個具名的變數,是否會對傳回值產生修改作用。
package mainimport ( "fmt")func main() { fmt.Println(test())}//輸出為0func test() int { var a int defer func() { a++ }() return a }
證明不能對傳回值進行修改了,具體跟範圍應該有關係,有時間再深究。當然如果將a返回給具名的傳回值result然後修改result當然能修改傳回值。
項目中遇到過的使用
在對panic的處理中,將panic的錯誤資訊作為傳回值,簡化版代碼如下:
package mainimport ( "fmt" "runtime/debug")func main() { fmt.Println(test())}func test() (err error) { defer func() { if e := recover(); e != nil { err = fmt.Errorf("%s \n panic:\n %s", e, debug.Stack()) //列印e(panic的異常資訊補貨結果),並且強烈推薦列印堆棧資訊用來定位錯誤。 } }() var a int a = 1 / a return nil}
返回結果如下:
runtime error: integer divide by zero panic: goroutine 1 [running]:runtime/debug.Stack(0xc042031d88, 0x49d220, 0xc042004060) C:/Go/src/runtime/debug/stack.go:24 +0x80main.test.func1(0xc042031ea8) C:/Users/colinhome/Documents/GitHub/test/src/structindex/base64.go:14 +0x6apanic(0x49d220, 0xc042004060) C:/Go/src/runtime/panic.go:458 +0x251main.test(0x0, 0x0) C:/Users/colinhome/Documents/GitHub/test/src/structindex/base64.go:19 +0x80main.main() C:/Users/colinhome/Documents/GitHub/test/src/structindex/base64.go:9 +0x3b
能定位到具體行數。方便調試。