learned! Use buffered channels as Mutexes
As part of the official Go package, the sync package has the following statement:
The sync package provides basic synchronization primitives such as mutex locks. Except for the Once and WaitGroup types, most other types are intended for the underlying function library. Higher level synchronization is better accomplished through channels and communications.
In the vast majority of examples you can find of allowing concurrent access, many use mutexes to solve the problem. However, there are few examples showing us how to use channels to provide synchronization mechanisms. So, let’s discuss it in this article.
Characteristics of mutex locks
In order for the mutex lock to work, you need to lock it when accessing the shared variable, and you need to unlock it after the operation is completed. . The same mutex is not allowed to be locked multiple times to avoid race conditions.
Unbuffered channel and its shortcomings
If there is no receiver, the sender will block; similarly, if there is no sender , the receiver will block. Based on this characteristic, we cannot use unbuffered channels as locks.
Let's see if the buffer channel can be used as a mutex lock.
Characteristics and advantages of a channel with a buffer size of 1
A channel with a buffer size of 1 has the following characteristics: If the buffer is full , it will be blocked when sending; if the cache is vacated, it will be unblocked when sending.
Obviously, the blocking characteristics of this channel are desirable. Compare it with the characteristics of mutex locks:
Lock when the buffer is full<-->
Buffer Vacation<--> Unlock
Let’s demonstrate this feature through code.
Demonstration: How to use the buffer channel as a "lock"
We assume that there is a column of names that need to be written to the file, and each name needs to be consecutive Write 1000 times, and do not allow overlap between different names.
package main import ( "errors" "fmt" "os" "sync" ) func main() { file, err := os.Create("record.txt") defer func() { if err := recover(); err != nil { fmt.Printf("Error encounter: %w", err) } file.Close() }() if err != nil { panic(errors.New("Cannot create/open file")) } ss := []string{ //string slice literals "James", "Avery", "Peter", "John", "Beau", } chanLock := make(chan int, 1) //1 var wg sync.WaitGroup for _, str := range ss { //2 wg.Add(1) //amended thanks to response from Wang //Sheng go func(aString string) { chanLock <- 1 //3 for i := 0; i < 1000; i++ { file.WriteString(aString + "\n") } <-chanLock //4 wg.Done() //5 }(str) //pass by value } wg.Wait() }
In the above code, //1 we created a channel with a buffer of 1. //2 We created the same number of goroutines as the number of names. //3 is equivalent to locking, //4 is equivalent to unlocking, so that multiple goroutines can write names to the record.txt file synchronously, but only one goroutine will operate the file at a time.
It should be noted that we use WaitGroup to ensure that the main coroutine will not exit before the sub-goroutine completes the task.
Hope this article is helpful to you, enjoy coding!
The above is the detailed content of learned! Use buffered channels as Mutexes. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics



Queue threading problem in Go crawler Colly explores the problem of using the Colly crawler library in Go language, developers often encounter problems with threads and request queues. �...

The library used for floating-point number operation in Go language introduces how to ensure the accuracy is...

The problem of using RedisStream to implement message queues in Go language is using Go language and Redis...

The difference between string printing in Go language: The difference in the effect of using Println and string() functions is in Go...

What should I do if the custom structure labels in GoLand are not displayed? When using GoLand for Go language development, many developers will encounter custom structure tags...

Two ways to define structures in Go language: the difference between var and type keywords. When defining structures, Go language often sees two different ways of writing: First...

Which libraries in Go are developed by large companies or well-known open source projects? When programming in Go, developers often encounter some common needs, ...

When using sql.Open, why doesn’t the DSN report an error? In Go language, sql.Open...
