首页 > 后端开发 > Golang > 如何在 Go 中正确使用'defer”和修改变量?

如何在 Go 中正确使用'defer”和修改变量?

Mary-Kate Olsen
发布: 2024-12-11 09:54:14
原创
460 人浏览过

How to Correctly Use `defer` in Go with Modified Variables?

Defer 用法说明

在 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 的值将在调用匿名函数时捕获,确保打印最新的值。

替代解决方案

  1. 使用指针:

    var x int
    
    defer func() { fmt.Println(&x) }()
    登录后复制

    此方法使用指向 x 的指针作为延迟函数的参数。当执行“defer”语句时,仅计算指针,而不计算 x 的值。当延迟函数被调用时,它将通过指针访问 x 的当前值。

  2. 使用自定义类型:

    type MyInt int
    
    func (m *MyInt) String() string {
      return strconv.Itoa(int(*m))
    }
    
    var x MyInt
    
    defer fmt.Println(&x)
    登录后复制

    此解决方案类似于指针方法,但使用实现 String() 方法的自定义类型 (MyInt)。通过实现 String(),我们可以控制 x 的值如何打印。

  3. 使用切片:

    var x []int
    
    defer fmt.Println(x)
    登录后复制

    切片是Go 中的描述符类型,这意味着它的值是对底层数组的引用。当我们延迟切片时,仅评估引用,而不评估实际的数组元素。因此,延迟后对切片元素所做的任何更改都将反映在打印输出中。

  4. 包装在结构中:

    type Wrapper struct {
      Value int
    }
    
    var x Wrapper
    
    defer fmt.Println(&x)
    登录后复制

    这种方法类似于使用指针,但我们将值包装在结构中以避免在结构体中取消引用指针延迟函数。

以上是如何在 Go 中正确使用'defer”和修改变量?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板