package main
import "fmt"
func main() {
fmt.Println("return a ", a())
fmt.Println("---------------------------------------")
b1,b2 := b()
fmt.Println("return b ", b1, b2)
fmt.Println("---------------------------------------")
c1,c2 := c()
fmt.Println("return c ", c1, c2)
fmt.Println("---------------------------------------")
}
//有名返回值
func a() (i int) {
defer func() {
i++
fmt.Println("defer a 2:", i)
}()
defer func() {
i++
fmt.Println("defer a 1", i)
}()
return i
}
//无名返回值
func b() (t *int, m int) {
var i int
defer func() {
i++
fmt.Println("defer b 2:", &i, i)
}()
defer func() {
i++
fmt.Println("defer b 1", &i, i)
}()
return &i,i
}
func c() (*int,int) {
var i int
defer func() {
i++
fmt.Println("defer c 2:", &i, i)
}()
defer func() {
i++
fmt.Println("defer c 1", &i, i)
}()
return &i,i
}
运行结果:
defer a 1 1
defer a 2: 2
return a 2
---------------------------------------
defer b 1 0x41602c 1
defer b 2: 0x41602c 2
return b 0x41602c 0
---------------------------------------
defer c 1 0x416048 1
defer c 2: 0x416048 2
return c 0x416048 0
---------------------------------------
分析结果:
func a是有名返回值函数
func b是无名返回值函数
func c是指针类型返回值函数
从这三个函数来看,defer函数的执行是在return 函数之后才执行,如果函数的返回值是无名的,则go语言会在执行return的时候会执行一个类似创建一个零时变量作为保存return值的动作,而有名返回值的函数,由于返回值在函数定义的时候已经将该变量进行定义,在执行return的时候会先执行返回值保存操作,而后续的defer函数会改变这个返回值(虽然defer是在return之后执行的,但是由于使用的函数定义的变量,所以执行defer操作后对该变量的修改会影响到return的值)