defer(4)

下面代码输出什么?

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}
}

答案解析:

29 28 28。

这道题之前题目的基础上做了一点点小改动,前一题最后一行代码 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。

以上解析来自 Go语言中文网


defer(4)
http://example.com/2023/07/15/Go每日一题/defer-5/
作者
Feng Tao
发布于
2023年7月15日
更新于
2023年7月15日
许可协议