首页 > 后端开发 > Golang > 为什么我的 Go 代码在 Goroutine 数据争用中打印了三次'三”而不是'一”、'二”和'三”?

为什么我的 Go 代码在 Goroutine 数据争用中打印了三次'三”而不是'一”、'二”和'三”?

Linda Hamilton
发布: 2024-12-07 22:23:13
原创
908 人浏览过

Why Does My Go Code Print

理解数据竞争场景中 Goroutine 的奇怪行为

在提供的 Go 代码中,我们创建了一个名为 data 的结构体字段切片包含名称“一”、“二”和“三”。该代码迭代切片,创建 goroutine 以使用 print 方法打印每个字段的名称。然而,与预期相反,代码重复打印“三”三次,而不是预期的“一”、“二”和“三”顺序。

揭开数据争用

这种奇怪的行为源于数据竞争,当多个 goroutine 访问并可能同时修改相同的共享数据时就会发生这种情况。在这种情况下,问题是由于创建 goroutine 时隐式使用范围变量 v 的地址而引起的。当循环变量 v 在每次迭代中更改时,goroutine 最终会使用其最终值,从而导致不断打印“三”。

解决数据争用

为了解决这个问题,我们可以采用几种方法:

  • 在每次循环迭代中创建一个新变量:在循环中,我们可以声明一个与范围变量同名的新变量,从而有效地创建变量的新范围。

    for _, v := range data {
      v := v // Declare a new variable `v` within the loop scope.
      go v.print()
    }
    登录后复制
  • 使用指针切片: 我们可以使用指向字段的指针切片,而不是使用结构体字段切片。这可以确保 goroutine 接收指向各个字段元素的指针,从而防止数据竞争问题。

    data := []*field{ {"one"},{"two"},{"three"} }
    for _, v := range data {
      go v.print()
    }
    登录后复制
  • 传递切片元素的地址: 另一种替代方法是将切片中每个元素的地址传递给goroutine.

    for i := range data {
      v := &data[i] // Take the address of the slice element.
      go v.print()
    }
    登录后复制
  • 使用匿名函数并传递范围变量作为参数:如果 goroutine 函数位于匿名函数内,我们可以通过传递来避免此问题范围变量作为参数

    for _, v := range data {
      go func(v field) { // Pass the range variable `v` as an argument.
          v.print()
      }(v)
    }
    登录后复制

这些方法确保 goroutine 拥有自己所需数据的副本,消除数据竞争并生成“一”、“二”的正确输出, ”和“三”,顺序不限。

以上是为什么我的 Go 代码在 Goroutine 数据争用中打印了三次'三”而不是'一”、'二”和'三”?的详细内容。更多信息请关注PHP中文网其他相关文章!

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