为了账号安全,请及时绑定邮箱和手机立即绑定

指针接收机和非指针接收机的数据竞争差异

指针接收机和非指针接收机的数据竞争差异

慕斯王 2022-08-01 16:16:54
我在使用标志进行测试期间发现了一场数据竞赛。在 更新结构并从结构方法读取值时,会发生数据争用。后来我发现将方法从非指针接收器改为指针接收器可以解决数据竞赛问题。但我不明白原因。谁能解释原因?-racepackage mainimport (    "fmt"    "testing")type TestStruct struct {    display    bool    OtherValue int}func (t TestStruct) Display() bool {    return t.display}func (t *TestStruct) DisplayP() bool {    return t.display}func TestNonPointerRecevier(t *testing.T) {    v := &TestStruct{        display: true,    }    go func() {        v.OtherValue = 1    }()    go func() {        fmt.Println(v.Display())    }()}func TestPointerRecevier(t *testing.T) {    v := &TestStruct{        display: true,    }    go func() {        v.OtherValue = 1    }()    go func() {        fmt.Println(v.DisplayP())    }()}使用指针接收器方法,您没有错误go test -race -run ^TestPointerRecevier$truePASSok      _/Users/xxxxx/projects/golang/datarace  0.254s使用非指针接收器方法时收到此错误go test -race -run ^TestNonPointerRecevier$==================WARNING: DATA RACERead at 0x00c00001c2c8 by goroutine 9:  _/Users/xxxxx/projects/golang/datarace.TestNonPointerRecevier.func2()      /Users/xxxxx/projects/golang/datarace/main_test.go:30 +0x47Previous write at 0x00c00001c2c8 by goroutine 8:  _/Users/xxxxx/projects/golang/datarace.TestNonPointerRecevier.func1()      /Users/xxxxx/projects/golang/datarace/main_test.go:27 +0x3eGoroutine 9 (running) created at:  _/Users/xxxxx/projects/golang/datarace.TestNonPointerRecevier()      /Users/xxxxx/projects/golang/datarace/main_test.go:29 +0xba  testing.tRunner()      /usr/local/Cellar/go/1.15.6/libexec/src/testing/testing.go:1123 +0x202Goroutine 8 (finished) created at:  _/Users/xxxxx/projects/golang/datarace.TestNonPointerRecevier()      /Users/xxxxx/projects/golang/datarace/main_test.go:26 +0x98  testing.tRunner()      /usr/local/Cellar/go/1.15.6/libexec/src/testing/testing.go:1123 +0x202==================trueFAILexit status 1FAIL    _/Users/xxxxx/projects/golang/datarace  0.103s
查看完整描述

2 回答

?
慕哥9229398

TA贡献1877条经验 获得超6个赞

当方法的接收器值是结构(而不是指向结构的指针)时,将复制完整的结构以按值传递给该方法。
因此,调用隐式读取字段(在创建结构的副本时),因此存在争用条件。v.Display()OtherValue

另一方面,使用指针时,仅复制指针,并且 并发访问 不会触发争用条件。v.displayv.OtherValue


查看完整回答
反对 回复 2022-08-01
?
红糖糍粑

TA贡献1815条经验 获得超6个赞


import (

    "fmt"

    "testing"

)


type TestStruct struct {

    display    bool

    OtherValue int

}


func Display(t TestStruct) bool { // equal func (t TestStruct) Display() bool

    return t.display

}


func DisplayP(t *TestStruct) bool { // equal func (t *TestStruct) DisplayP() bool

    return t.display

}


func TestNonPointerRecevier(t *testing.T) {

    v := &TestStruct{

        display: true,

    }


    go func() {

        v.OtherValue = 1 // write

    }()

    go func() {

        fmt.Println(Display(*v)) // *v read value

    }()

}


func TestPointerRecevier(t *testing.T) {

    v := &TestStruct{

        display: true,

    }


    go func() {

        v.OtherValue = 1 // write

    }()

    go func() {

        fmt.Println(DisplayP(v))  // un read, just pass parameter

    }()

}

关注数据竞跑错误信息,调用方法时已经发生过,go会将v()转换为()。readDisplaytype *TestStructtype TestStruct


查看完整回答
反对 回复 2022-08-01
  • 2 回答
  • 0 关注
  • 141 浏览

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号