在 Go 中,“defer”语句允许我们安排一个函数在周围函数返回之前执行。但是,在处理在封闭函数中修改的变量时,这可能会导致混乱。
考虑以下函数:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
根据Go规范中,当执行“defer”语句时,函数值和参数将被评估并存储以供以后执行。这意味着当函数最终被调用时,x 的值仍将为 0,因为它是在延迟时计算的。
为了解决这个问题,我们可以在“defer”语句内使用匿名函数:
defer func() { fmt.Println(x) }()
这里,x不是匿名函数的参数,所以不会当执行“defer”语句时评估。相反,x 的值将在调用匿名函数时捕获,确保打印最新的值。
使用指针:
var x int defer func() { fmt.Println(&x) }()
此方法使用指向 x 的指针作为延迟函数的参数。当执行“defer”语句时,仅计算指针,而不计算 x 的值。当延迟函数被调用时,它将通过指针访问 x 的当前值。
使用自定义类型:
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } var x MyInt defer fmt.Println(&x)
此解决方案类似于指针方法,但使用实现 String() 方法的自定义类型 (MyInt)。通过实现 String(),我们可以控制 x 的值如何打印。
使用切片:
var x []int defer fmt.Println(x)
切片是Go 中的描述符类型,这意味着它的值是对底层数组的引用。当我们延迟切片时,仅评估引用,而不评估实际的数组元素。因此,延迟后对切片元素所做的任何更改都将反映在打印输出中。
包装在结构中:
type Wrapper struct { Value int } var x Wrapper defer fmt.Println(&x)
这种方法类似于使用指针,但我们将值包装在结构中以避免在结构体中取消引用指针延迟函数。
以上是如何在 Go 中正确使用'defer”和修改变量?的详细内容。更多信息请关注PHP中文网其他相关文章!