目录

defer语句

defer语句

defer及defer函数的执行顺序分2步:

  1. 执行defer语句,计算函数的入参的值,并传递给函数,但不执行函数,而是将函数压入栈。
  2. 函数return语句后,或panic后,执行压入栈的函数,函数中变量的值,此时会被计算。

Each time a “defer” statement executes, the function value and parameters to the call are evaluated as usual and saved anew but the actual function is not invoked.

defer语句的引用问题

注意: defer 后面常跟匿名函数,函数内用到外面的变量,属于闭包,变量是引用

func main() {
	fmt.Println(test()) //5
	fmt.Println(test2()) //4
}

func test()(x int){
	x=2
	defer func() {
		x=5
		fmt.Println(x)//5
	}()
	return 4
}

func test2()(x int){
	x=2
	defer func(i int) {
		i=5
	}(x)
	return 4
}

defer与return

编译器将 defer 处理成两个函数调用,deferproc 定义一个延迟调用对象,然后在函数结束 前通过 deferreturn 完成最终调用。

大致表达为: step 1 : 在defer表达式的地方,会调用runtime.deferproc(size int32, fn *funcval)保存延时调用,注意这里保存了延时调用的参数 step 2 : 在return时,先将返回值保存起来 step 3 : 按FILO顺序调用runtime.deferreturn,即延时调用 step 4 : RET指令

func namedReturn() (r int) {
    defer func() {
        r++
        fmt.Println("defer in namedReturn : r = ", r)
    }()

    return
}

func unnamedReturn() int {
    var r int
    defer func() {
        r++
        fmt.Println("defer in unnamedReturn : r = ", r)
    }()
    return r
}

func main() {
	//1
    fmt.Println("namedReturn : r = ", namedReturn())
    
	//0
    fmt.Println("unnamedReturn : r = ", unnamedReturn())
}

原因就是return会将返回值先保存起来,对于无名返回值来说,保存在一个临时对象中,defer是看不到这个临时对象的;而对于有名返回值来说,就保存在已命名的变量中。 匿名返回值是在return执行时被声明,有名返回值则是在函数声明的同时被声明,因此在defer语句中只能访问有名返回值,而不能直接访问匿名返回值;