go 函数并发编程中提供了三种锁类型:互斥锁(mutex)、读写锁(rwmutex)、一次性锁(once)。互斥锁保证独占访问,读写锁允许并发读取和单一写入,一次性锁确保代码块仅执行一次。这些锁用于协调对共享资源的访问,防止数据竞争。实践中,互斥锁可确保并发 web 服务中的数据一致性,阻止并发请求同时修改共享数据结构。
Go 函数并发编程中的锁类型和用法
在 Go 函数并发编程中,锁是一种同步机制,用于协调对共享资源的访问,防止数据竞争。Go 中提供了多种类型的锁,每种锁都有不同的特性和适用场景。
互斥锁
互斥锁(Mutex)是一个基础的锁,只能同时允许一个 goroutine 获取资源。它保证了对共享资源的独占访问。
import ( "sync" "fmt" ) var ( mu sync.Mutex counter int ) func main() { for i := 0; i < 1000; i++ { go func() { mu.Lock() counter++ mu.Unlock() }() } fmt.Println("Final counter value:", counter) }
登录后复制
读写锁
读写锁(RWMutex)允许多个 goroutine 并发读取共享资源,但只有一个 goroutine 可以同时写入资源。
import ( "sync" "fmt" ) var ( rwmu sync.RWMutex shared []int ) func main() { // 多个 goroutine 并发读取共享切片 for i := 0; i < 1000; i++ { go func() { rwmu.RLock() fmt.Println("Read:", shared) rwmu.RUnlock() }() } // 单独的 goroutine 写入共享切片 go func() { rwmu.Lock() shared = append(shared, 1, 2, 3) rwmu.Unlock() }() }
登录后复制
Once
Once 是一个一次性锁,用于确保特定代码块只执行一次。
import ( "sync" "fmt" ) var ( initOnce sync.Once inited = false ) func initialize() { inited = true fmt.Println("Initialized") } func main() { initOnce.Do(initialize) if inited { fmt.Println("Already initialized") } else { fmt.Println("Not initialized") } }
登录后复制
实战案例:确保并发 web 服务中的数据一致性
假设有一个 web 服务,其中多并发请求需要操作同一个共享数据结构。为了确保数据的一致性,可以使用互斥锁来保护数据结构,阻止并发请求同时对其进行修改。
import ( "sync" "net/http" ) var ( mu sync.Mutex clients map[string]*http.Client ) func main() { http.HandleFunc("/addClient", func(w http.ResponseWriter, r *http.Request) { mu.Lock() clients[r.FormValue("name")] = &http.Client{} mu.Unlock() }) }
登录后复制
在本例中,使用互斥锁 mu
保护对 clients
地图的并发访问,确保在同一时间只有一个请求可以添加或修改客户信息,从而避免了数据竞争。
以上就是Golang函数并发编程中锁的类型和用法的详细内容,更多请关注叮当号网其它相关文章!
文章来自互联网,只做分享使用。发布者:momo,转转请注明出处:https://www.dingdanghao.com/article/367095.html