>백엔드 개발 >Golang >Go 구조체를 동시에 읽고 쓸 때 데이터 경합을 방지하려면 어떻게 해야 합니까?

Go 구조체를 동시에 읽고 쓸 때 데이터 경합을 방지하려면 어떻게 해야 합니까?

Mary-Kate Olsen
Mary-Kate Olsen원래의
2024-12-25 04:21:09536검색

How Can I Prevent Data Races When Concurrently Reading and Writing Go Structs?

명시적 잠금 없이 Go 구조체의 동시 읽기 및 쓰기

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 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.