defer退出顺序

1
2
3
4
5
6
7
8
9
func f() {
defer fmt.Println("D")
fmt.Println("F")
}

func main() {
f()
fmt.Println("M")
}

被调用函数里的 defer 语句在返回之前就会被执行,所以输出顺序是 F D M。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
type Person struct {
age int
}

func main() {
person := &Person{28}

// 1.
defer fmt.Println(person.age)

// 2.
defer func(p *Person) {
fmt.Println(p.age)
}(person)

// 3.
defer func() {
fmt.Println(person.age)
}()

person = &Person{29}
}

这道题之前题目的基础上做了一点点小改动,前一题最后一行代码 person.age = 29 是修改引用对象的成员 age,这题最后一行代码 person = &Person{29} 是修改引用对象本身,来看看有什么区别。

  1. person.age 这一行代码跟之前含义是一样的,此时是将 28 当做 defer 函数的参数,会把 28 缓存在栈中,等到最后执行该 defer 语句的时候取出,即输出 28;

  2. defer 缓存的是结构体 Person{28} 的地址,这个地址指向的结构体没有被改变,最后 defer 语句后面的函数执行的时候取出仍是 28;

  3. 闭包引用,person 的值已经被改变,指向结构体 Person{29},所以输出 29.

由于 defer 的执行顺序为先进后出,即 3 2 1,所以输出 29 28 28。

作者

JIeJaitt

发布于

2023-11-03

更新于

2023-11-13

许可协议

Your browser is out-of-date!

Update your browser to view this website correctly.&npsb;Update my browser now

×