Defer 用法说明:打印最终变量值
在 Go 中,defer 关键字允许我们安排一个函数在周围函数返回。但是,如果延迟函数使用可能在周围函数内更改的局部变量,则必须特别注意确保使用正确的值。
考虑以下函数:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
根据语言规范,当 defer 语句执行时,局部变量会被重新求值并保存。然而,实际的函数直到稍后才会被调用。在这种情况下,当执行 defer 语句时,x 的值为 0,因此调用 deferred 函数时将打印 0。
要解决此问题,我们可以使用以下几种方法之一:
1。使用匿名函数:
defer func() { fmt.Println(x) }()
这使用捕获 x 的当前值的匿名函数。当匿名函数被调用时,它将使用捕获的值,即使 x 在周围函数中发生了变化。
2.使用指针:
var x int defer Print(&x)
我们可以使用指向 x 的指针并延迟打印指向值的函数。这是可行的,因为执行 defer 语句时会计算指针值,但直到调用 deferred 函数时才会计算指向的值。
3.使用自定义类型:
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } var x MyInt defer fmt.Println(&x)
在这里,我们创建一个实现 fmt.Stringer 接口的自定义类型 MyInt,允许我们指定如何打印值。当 &x 被打印时,将调用 String 方法,返回指向值的字符串表示形式。
4.包装:
x := []int{0} defer fmt.Println(x)
将变量包装在切片中允许我们推迟打印切片内容的函数。执行 defer 语句时会计算切片描述符,但直到调用延迟函数时才会计算底层数组。
请务必记住,延迟函数中使用的变量不应该是延迟函数本身,因为参数在延迟语句执行时被评估。
以上是Go中处理局部变量变化时如何正确使用Defer?的详细内容。更多信息请关注PHP中文网其他相关文章!