Golang中鎖的原理及應用解析
2.1 互斥鎖
互斥鎖的基本原理是透過一個標誌位元來表示資源是否被鎖定。當某個 goroutine 要存取被互斥鎖保護的資源時,首先會嘗試取得鎖。如果鎖已經被其他 goroutine 獲取,那麼當前 goroutine 就會被阻塞,直到鎖被釋放。當一個 goroutine 取得到鎖之後就可以對資源進行操作,並且在完成操作後釋放鎖,以便其他 goroutine 取得鎖。
下面是互斥鎖的應用範例:
package main import ( "fmt" "sync" ) var count int var lock sync.Mutex func increase() { for i := 0; i < 100000; i++ { lock.Lock() count++ lock.Unlock() } } func main() { var wg sync.WaitGroup for i := 0; i < 10; i++ { wg.Add(1) go func() { increase() wg.Done() }() } wg.Wait() fmt.Println("count:", count) }
在上述範例中,我們建立了一個互斥鎖定lock
來保護全域變數count
的讀寫操作。然後啟動了10個 goroutine 來並發操作 count
變量,每個 goroutine 都會對 count
執行 100000 次加操作。最後輸出 count
的值,我們會發現結果總是 1000000,表示互斥鎖確實保證了並發操作的正確性。
2.2 讀寫鎖定
讀寫鎖定是互斥鎖的擴展,它允許多個 goroutine 同時讀取共享資源,並且保證在寫入資源時只能有一個 goroutine。讀寫鎖定可以提高並發處理讀取操作的效率。
下面是讀寫鎖定的應用範例:
package main import ( "fmt" "sync" "time" ) var count int var rwlock sync.RWMutex func read() { rwlock.RLock() defer rwlock.RUnlock() fmt.Println("read:", count) } func write() { rwlock.Lock() defer rwlock.Unlock() count = count + 1 fmt.Println("write:", count) } func main() { go read() go write() time.Sleep(time.Second) }
在上述範例中,我們建立了一個讀寫鎖定rwlock
來保護全域變數count
的讀寫操作。啟動了兩個 goroutine,其中一個進行讀取操作,另一個進行寫入操作。由於讀取操作可以同時進行,因此讀取操作會先執行,輸出 read:
和 write:
的順序可能會不一樣。但是我們可以確保寫入操作先執行,然後再進行讀取操作,以確保資料的正確性。
以上是深入解析Golang中鎖的原理與應用的詳細內容。更多資訊請關注PHP中文網其他相關文章!