Go에서 공유 데이터에 대한 동시 액세스는 데이터 경합과 같은 잠재적인 오류의 원인이 될 수 있습니다. 데이터 구조에 대한 동시 액세스, 즉 여러 고루틴이 동시에 액세스할 수 있는 경우 불일치를 방지하기 위해 데이터를 동기화된 방식으로 읽고 쓰는 것이 중요합니다.
다음 Go 구조체 메타데이터를 고려하세요.
type Metadata struct { mu sync.RWMutex // ? key bool }
가능한 한 메타데이터 구조체에는 bool 유형의 필드 키와 읽기-쓰기 잠금 구현인 sync.RWMutex 유형의 또 다른 필드 mu가 포함되어 있습니다.
메타데이터 인스턴스를 생성하고 여러 고루틴이 해당 필드를 동시에 읽고 쓸 수 있도록 허용하면 데이터 경합이 발생할 수 있습니다. 여러 고루틴이 동일한 데이터에 동시에 액세스하고 그 중 적어도 하나가 쓰기 작업을 수행할 때 데이터 경합이 발생합니다.
다음 코드는 명시적인 잠금 없이 메타데이터 구조체에 대한 동시 읽기 및 쓰기 액세스를 보여줍니다.
func concurrentStruct() { m := new(Metadata) for i := 0; i < 100000; i++ { go func(metadata *Metadata) { for { readValue := metadata.key if readValue { metadata.key = false } } }(m) go func(metadata *Metadata) { for { metadata.key = true } }(m) } select {} }
이 코드에서는 키 필드를 동시에 읽고 쓰는 고루틴을 만듭니다. 우리는 select 문을 사용하여 기본 고루틴을 차단하여 동시 고루틴이 실행될 수 있도록 합니다. go run -race 명령을 사용하여 프로그램을 실행하면 DATA RACE를 나타내는 경고가 표시됩니다.
그러나 프로그램은 충돌 없이 계속 실행됩니다. 이는 Go 런타임에 동시성 검사 기능이 내장되어 있지만 안전한 실행을 보장하지 않기 때문입니다. 이 경우 데이터 경합으로 인해 정의되지 않은 동작과 잘못된 결과가 발생할 수 있습니다.
구조체에 동시에 읽고 쓸 때 데이터 경합을 방지하려면 적절한 사용 방법을 사용해야 합니다. 잠금 메커니즘. 한 가지 방법은 다음 코드에 설명된 대로 뮤텍스를 사용하는 것입니다.
func concurrentStructWithMuLock() { m := new(Metadata) go func(metadata *Metadata) { for { metadata.mu.Lock() readValue := metadata.key if readValue { metadata.key = false } metadata.mu.Unlock() } }(m) go func(metadata *Metadata) { for { metadata.mu.Lock() metadata.key = true metadata.mu.Unlock() } }(m) select {} }
이 코드에서는 메타데이터 구조체에 읽기-쓰기 잠금을 추가하고 mu.Lock() 및 mu를 사용합니다. Unlock()은 키 필드에 대한 액세스를 동기화합니다. go run -race를 사용하여 프로그램을 실행하면 더 이상 데이터 경합이 없음을 나타내는 경고가 생성되지 않습니다.
위 내용은 Go 구조체를 동시에 읽고 쓸 때 데이터 경합을 방지하려면 어떻게 해야 합니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!