go中是先给return准备返回值,再根据defer先进后出的规则执行,最后将返回值返回给调用者
代码片段如下:
func foo_1() (err error) {
defer func() {
fmt.Println(err)
err = errors.New("a")
}()
defer func(e error) {
fmt.Println(e)
e = errors.New("b")
}(err)
err = errors.New("c")
return err
}
func TestDeferAndReturn(t *testing.T) {
fmt.Println(foo_1())
}
分析:
这个测试用例的输出如下
=== RUN TestDeferAndReturn
c
a
--- PASS: TestDeferAndReturn (0.00s)
PASS
Process finished with exit code 0
代码片段如下:
func foo_2() error {
var err error
defer func() {
fmt.Println(err)
err = errors.New("a")
}()
defer func(e error) {
fmt.Println(e)
e = errors.New("b")
}(err)
err = errors.New("c")
return err
}
func TestDeferAndReturn(t *testing.T) {
fmt.Println(foo_2())
}
分析:
这里的返回值是匿名返回写法,这种情况下,return会首先创建一个临时变量temp,然后将err赋值给temp,后续两个defer中对err的操作并不会影响到temp中的值,所以这里返回给调用者的是c
这个测试用例的输出如下
=== RUN TestDeferAndReturn
c
c
--- PASS: TestDeferAndReturn (0.00s)
PASS
Process finished with exit code 0
原文:https://www.cnblogs.com/Kingram/p/14548529.html