理解数据竞争场景中 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中文网其他相关文章!