Golang의 동기화 메커니즘 및 성능 병목 현상 최적화 솔루션
package main import ( "fmt" "sync" ) var ( count int lock sync.Mutex ) func increment() { lock.Lock() defer lock.Unlock() count++ } func main() { var wg sync.WaitGroup for i := 0; i < 1000; i++ { wg.Add(1) go func() { defer wg.Done() increment() }() } wg.Wait() fmt.Println(count) }
2.2 조건 변수(Cond)
조건 변수는 코루틴 간 통신을 수행하고 코루틴의 대기 및 깨우기 메커니즘을 구현하는 데 사용됩니다. 코루틴이 특정 조건을 충족하면 조건 변수를 사용하여 다른 코루틴에 알릴 수 있습니다. 다음은 조건 변수를 사용한 샘플 코드입니다.
package main import ( "fmt" "sync" ) var ( ready bool cond *sync.Cond ) func init() { cond = sync.NewCond(&sync.Mutex{}) } func printNumbers() { cond.L.Lock() defer cond.L.Unlock() for !ready { cond.Wait() } fmt.Println("1 2 3 4 5") } func main() { go printNumbers() cond.L.Lock() ready = true cond.Signal() cond.L.Unlock() }
2.3 읽기-쓰기 잠금(RWMutex)
읽기-쓰기 잠금은 동시 프로그램의 성능을 더욱 향상시킬 수 있습니다. 읽기는 많고 쓰기는 적은 시나리오에서 읽기-쓰기 잠금을 사용하면 여러 코루틴이 동시에 공유 리소스를 읽을 수 있지만 하나의 코루틴만 쓰기 작업을 수행할 수 있습니다. 다음은 읽기-쓰기 잠금을 사용하는 샘플 코드입니다.
package main import ( "fmt" "sync" ) var ( count int lock sync.RWMutex ) func read() { lock.RLock() defer lock.RUnlock() fmt.Println(count) } func write() { lock.Lock() defer lock.Unlock() count++ } func main() { var wg sync.WaitGroup wg.Add(10) for i := 0; i < 5; i++ { go func() { defer wg.Done() read() }() go func() { defer wg.Done() write() }() } wg.Wait() }
3.1 잠금 세분성 줄이기
뮤텍스 잠금을 사용할 때 잠금 세분성을 최대한 줄이고 필요한 중요 섹션 코드 세그먼트만 잠글 수 있습니다. 이렇게 하면 잠금 경합이 줄어듭니다. 읽기-쓰기 잠금을 사용할 때 실제 상황에 따라 읽기 잠금 또는 쓰기 잠금을 선택하여 병렬 읽기의 특성을 최대한 활용할 수 있습니다.
3.2 잠금 없는 데이터 구조 사용
동시성이 높은 시나리오의 경우 원자 패키지의 원자 연산 기능과 같은 잠금 없는 데이터 구조 사용을 고려할 수 있습니다. 이러한 함수는 데이터 일관성을 보장하기 위해 잠금을 사용하지 않고 일부 원자성 작업을 제공합니다. 예를 들어, 일관된 계산을 보장하려면 뮤텍스 대신에omic.AddInt64()를 사용하십시오.
3.3 뮤텍스 잠금 대신 채널 사용
채널은 데이터 액세스의 순서와 일관성을 보장하기 위한 동기화 메커니즘으로 사용될 수 있습니다. 일부 시나리오에서는 채널을 사용하면 뮤텍스 잠금의 명시적인 사용을 방지하여 잠금 경합을 줄일 수 있습니다. 그러나 차단이나 메모리 누수를 방지하려면 채널 용량과 성능 오버헤드에 주의를 기울여야 합니다.
위 내용은 Golang의 동기화 메커니즘 및 성능 병목 현상 최적화 솔루션의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!