gpt4 book ai didi

go - 当对同一个变量调用两次 defer 时会发生什么?

转载 作者:IT老高 更新时间:2023-10-28 13:04:16 25 4
gpt4 key购买 nike

当该方法的结构被更改时,defer 被调用两次时发生了什么?

例如:

rows := Query(`SELECT FROM whatever`)
defer rows.Close()
for rows.Next() {
// do something
}
rows = Query(`SELECT FROM another`)
defer rows.Close()
for rows.Next() {
// do something else
}

最后一个 rows.Close() 调用时是哪个 rows

最佳答案

这取决于方法接收器变量的类型。

简短回答:如果您使用的是 database/sql 包裹,您的延期 Rows.Close() 方法将正确关闭您的 Rows实例因为 Rows.Close()指针接收器因为 DB.Query() 返回一个指针(所以rows是一个指针)。请参阅下面的推理和解释。

为避免混淆,我建议使用不同的变量,这将清楚您想要什么以及将要关闭什么:

rows := Query(`SELECT FROM whatever`)
defer rows.Close()
// ...
rows2 := Query(`SELECT FROM whatever`)
defer rows2.Close()

我想指出一个重要的事实,即延迟函数及其参数被立即评估,这在 Effective Go 中有所说明。博客文章和 Language Spec: Deferred statements太:

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. Instead, deferred functions are invoked immediately before the surrounding function returns, in the reverse order they were deferred.

如果变量不是指针:调用延迟方法时会观察到不同的结果,具体取决于方法是否有指针接收器。
如果变量是指针,您将始终看到“想要的”结果。

看这个例子:

type X struct {
S string
}

func (x X) Close() {
fmt.Println("Value-Closing", x.S)
}

func (x *X) CloseP() {
fmt.Println("Pointer-Closing", x.S)
}

func main() {
x := X{"Value-X First"}
defer x.Close()
x = X{"Value-X Second"}
defer x.Close()

x2 := X{"Value-X2 First"}
defer x2.CloseP()
x2 = X{"Value-X2 Second"}
defer x2.CloseP()

xp := &X{"Pointer-X First"}
defer xp.Close()
xp = &X{"Pointer-X Second"}
defer xp.Close()

xp2 := &X{"Pointer-X2 First"}
defer xp2.CloseP()
xp2 = &X{"Pointer-X2 Second"}
defer xp2.CloseP()
}

输出:

Pointer-Closing Pointer-X2 Second
Pointer-Closing Pointer-X2 First
Value-Closing Pointer-X Second
Value-Closing Pointer-X First
Pointer-Closing Value-X2 Second
Pointer-Closing Value-X2 Second
Value-Closing Value-X Second
Value-Closing Value-X First

Go Playground 上试用.

使用指针变量,结果总是好的(如预期的那样)。

使用非指针变量和指针接收器我们看到相同的打印结果(最新),但如果我们有值接收器,它会打印 2 个不同的结果。

非指针变量说明:

如上所述,包含接收器的延迟函数在 defer 时进行评估。执行。如果是指针接收器,它将是局部变量的地址。因此,当您为其分配一个新值并调用另一个 defer ,指针接收器将再次是局部变量的相同地址(只是指向的值不同)。所以稍后在执行函数时,两者都会使用相同的地址两次,但 pointed 的值将是相同的,即稍后分配的那个。

如果是值接收器,接收器是一个副本,它是在 defer 时制作的。执行,所以如果你给变量赋值并调用另一个 defer , 将制作另一份与前一份不同的副本。

关于go - 当对同一个变量调用两次 defer 时会发生什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28893586/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com