Go 语言为什么不支持并发读写 map

2024-01-03 13:16:48

Go语言的map类型不支持并发读写的主要原因是并发读写会导致数据竞态(data race),这意味着多个 goroutine 可能同时访问并修改同一个 map,从而引发不确定的结果。

在Go语言的设计中,为了防止数据竞态,不同于一些其他语言,map并没有提供内置的锁机制。这样设计的目的是为了鼓励开发者使用更加精细的同步措施,以适应不同的并发场景。
?
小插曲 白嫖Goland激活码获取地址:https://web.52shizhan.cn

?
如果你需要在多个 goroutine 中安全地使用 map,可以考虑以下几种方法:

1 加锁: 使用 sync.Mutexsync.RWMutex 来保护对 map 的读写操作。

```go
package main

import (
    "sync"
)

var (
    mu   sync.Mutex
    data = make(map[string]string)
)

func writeToMap(key, value string) {
    mu.Lock()
    defer mu.Unlock()
    data[key] = value
}

func readFromMap(key string) string {
    mu.Lock()
    defer mu.Unlock()
    return data[key]
}

func main() {
    // 使用 writeToMap 和 readFromMap 安全地对 map 进行读写
}
```

2 使用 sync.Map 在Go 1.9及以上版本,标准库中提供了 sync.Map 类型,它是一种并发安全的 map 实现。

```go
package main

import (
    "sync"
)

var m sync.Map

func main() {
    // 使用 m.Store() 和 m.Load() 安全地对 map 进行读写
}
```

`sync.Map` 提供了一些方法来实现并发安全的读写操作,而无需额外的锁。

3 使用通道: 可以通过通道在不同的 goroutine 之间传递消息,避免直接对 map 进行并发访问。

```go
package main

import (
    "sync"
)

var (
    data     = make(map[string]string)
    readCh   = make(chan readRequest)
    writeCh  = make(chan writeRequest)
    shutdown = make(chan struct{})
    wg       sync.WaitGroup
)

type readRequest struct {
    key    string
    result chan<- string
}

type writeRequest struct {
    key, value string
}

func startDispatcher() {
    for {
        select {
        case req := <-readCh:
            req.result <- data[req.key]
        case req := <-writeCh:
            data[req.key] = req.value
        case <-shutdown:
            return
        }
    }
}

func writeToMap(key, value string) {
    writeCh <- writeRequest{key, value}
}

func readFromMap(key string) string {
    resultCh := make(chan string)
    readCh <- readRequest{key, resultCh}
    return <-resultCh
}

func main() {
    go startDispatcher()

    // 使用 writeToMap 和 readFromMap 安全地对 map 进行读写

    // 关闭通道和等待后台 goroutine 完成
    close(shutdown)
    wg.Wait()
}
```

这些方法中,具体选择取决于应用场景和需求。使用锁可能会引入一些开销,而 sync.Map 或基于通道的方法可能更适用于某些情况。

文章来源:https://blog.csdn.net/phpCenter/article/details/135359749
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。