package main
import "fmt"
//验证golang return defer 的返回值
func main() {
fmt.Println("let's go !!!")
i1 := test1()
fmt.Println("main1", i1, &i1)
fmt.Println("=============================")
i2 := test2()
fmt.Println("main2", i2, &i2)
}
func test1() int {
var i int
defer func() {
i += 1
fmt.Println("i is :", i, &i)
}()
defer func() {
i += 1
fmt.Println("i is :", i, &i)
}()
fmt.Println("test1", i, &i)
return i
}
func test2() (i int) {
defer func() {
i += 1
fmt.Println("i is :", i, &i)
}()
defer func() {
i += 1
fmt.Println("i is :", i, &i)
}()
fmt.Println("test2", i, &i)
return
}
结果:
let's go !!!
test1 0 0xc000088008
i is : 1 0xc000088008
i is : 2 0xc000088008
main1 0 0xc000088000
=============================
test2 0 0xc000018060
i is : 1 0xc000018060
i is : 2 0xc000018060
main2 2 0xc000018058
通过结果可以得出结论:
1.在test中的i指针和返回给main函数的i指针是不一样的,其实return是把i进行了一次赋值拷贝返回给main函数。
2.test1函数的返回值没有被提前声明,return返回值来自于函数内部定义变量(var i int)的赋值,而defer中修改的也是内部定义变量,defer无法直接访问到返回值,因此函数退出时返回值并没有被修改。
3.test2函数的返回值(i int)被提前声明,所以defer可以访问该返回值,因此在return赋值返回值 i 之后,defer调用返回值 i 并进行了修改,返回给main函数的值也就变的跟i一样。
package main
import (
"fmt"
"time"
)
//验证golang return defer 的返回值
func main() {
defer test(time.Now().Unix())
fmt.Println("let's go !!!")
time.Sleep(5 * time.Second)
fmt.Println("main timestamp is :", time.Now().Unix())
}
func test(t int64) {
fmt.Println("arg timestamp is :", t)
fmt.Println("test timestamp is :", time.Now().Unix())
}
结果:
let's go !!!
main timestamp is : 1597734408
arg timestamp is : 1597734403
test timestamp is : 1597734408
可以看到test函数中arg的时间戳的值比休眠之后的时间戳还小,说明defer test函数方法的参数time.Now().Unix()是先被调用了而不是等到执行defer的时候才调用。