自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Go 語言 Map 是并發(fā)安全的嗎?

開發(fā) 前端
在多個 goroutine 同時訪問同一個 map 時,可能會出現(xiàn)并發(fā)不安全的現(xiàn)象。這是因為 Go 語言中的 map 并沒有內(nèi)置鎖來保護對map的訪問。

Go 語言中的 map 是一個非常常用的數(shù)據(jù)結(jié)構(gòu),它允許我們快速地存儲和檢索鍵值對。然而,在并發(fā)場景下使用 map 時,還是有一些問題需要注意的。

本文將探討 Go 語言中的 map 是否是并發(fā)安全的,并提供三種方案來解決并發(fā)問題。

先來回答一下題目的問題,答案就是并發(fā)不安全。

看一段代碼示例,當兩個 goroutine 同時對同一個 map 進行寫操作時,會發(fā)生什么?

package main

import "sync"

func main() {
    m := make(map[string]int)
    m["foo"] = 1

    var wg sync.WaitGroup
    wg.Add(2)

    go func() {
        for i := 0; i < 1000; i++ {
            m["foo"]++
        }
        wg.Done()
    }()

    go func() {
        for i := 0; i < 1000; i++ {
            m["foo"]++
        }
        wg.Done()
    }()

    wg.Wait()
}

在這個例子中,我們可以看到,兩個 goroutine 將嘗試同時對 map 進行寫入。運行這個程序時,我們將看到一個錯誤:

fatal error: concurrent map writes

也就是說,在并發(fā)場景下,這樣操作 map 是不行的。

為什么是不安全的

因為它沒有內(nèi)置的鎖機制來保護多個 goroutine 同時對其進行讀寫操作。

當多個 goroutine 同時對同一個 map 進行讀寫操作時,就會出現(xiàn)數(shù)據(jù)競爭和不一致的結(jié)果。

就像上例那樣,當兩個 goroutine 同時嘗試更新同一個鍵值對時,最終的結(jié)果可能取決于哪個 goroutine 先完成了更新操作。這種不確定性可能會導(dǎo)致程序出現(xiàn)錯誤或崩潰。

Go 語言團隊沒有將 map 設(shè)計成并發(fā)安全的,是因為這樣會增加程序的開銷并降低性能。

如果 map 內(nèi)置了鎖機制,那么每次訪問 map 時都需要進行加鎖和解鎖操作,這會增加程序的運行時間并降低性能。

此外,并不是所有的程序都需要在并發(fā)場景下使用 map,因此將鎖機制內(nèi)置到 map 中會對那些不需要并發(fā)安全的程序造成不必要的開銷。

在實際使用過程中,開發(fā)人員可以根據(jù)程序的需求來選擇是否需要保證 map 的并發(fā)安全性,從而在性能和安全性之間做出權(quán)衡。

如何并發(fā)安全

接下來介紹三種并發(fā)安全的方式:

  1. 讀寫鎖
  2. 分片加鎖
  3. sync.Map

加讀寫鎖

第一種方法是使用讀寫鎖,這是最容易想到的一種方式。在讀操作時加讀鎖,在寫操作時加寫鎖。

package main

import (
    "fmt"
    "sync"
)

type SafeMap struct {
    sync.RWMutex
    Map map[string]string
}

func NewSafeMap() *SafeMap {
    sm := new(SafeMap)
    sm.Map = make(map[string]string)
    return sm
}

func (sm *SafeMap) ReadMap(key string) string {
    sm.RLock()
    value := sm.Map[key]
    sm.RUnlock()
    return value
}

func (sm *SafeMap) WriteMap(key string, value string) {
    sm.Lock()
    sm.Map[key] = value
    sm.Unlock()
}

func main() {
    safeMap := NewSafeMap()

    var wg sync.WaitGroup

    // 啟動多個goroutine進行寫操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            safeMap.WriteMap(fmt.Sprintf("name%d", i), fmt.Sprintf("John%d", i))
        }(i)
    }

    wg.Wait()

    // 啟動多個goroutine進行讀操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            fmt.Println(safeMap.ReadMap(fmt.Sprintf("name%d", i)))
        }(i)
    }

    wg.Wait()
}

在這個示例中,我們定義了一個 SafeMap 結(jié)構(gòu)體,它包含一個 sync.RWMutex 和一個 map[string]string。

定義了兩個方法:ReadMap 和 WriteMap。在 ReadMap 方法中,我們使用讀鎖來保護對 map 的讀取操作。在 WriteMap 方法中,我們使用寫鎖來保護對 map 的寫入操作。

在 main 函數(shù)中,我們啟動了多個 goroutine 來進行讀寫操作,這些操作都是安全的。

分片加鎖

上例中通過對整個 map 加鎖來實現(xiàn)需求,但相對來說,鎖會大大降低程序的性能,那如何優(yōu)化呢?其中一個優(yōu)化思路就是降低鎖的粒度,不對整個 map 進行加鎖。

這種方法是分片加鎖,將這個 map 分成 n 塊,每個塊之間的讀寫操作都互不干擾,從而降低沖突的可能性。

package main

import (
    "fmt"
    "sync"
)

const N = 16

type SafeMap struct {
    maps  [N]map[string]string
    locks [N]sync.RWMutex
}

func NewSafeMap() *SafeMap {
    sm := new(SafeMap)
    for i := 0; i < N; i++ {
        sm.maps[i] = make(map[string]string)
    }
    return sm
}

func (sm *SafeMap) ReadMap(key string) string {
    index := hash(key) % N
    sm.locks[index].RLock()
    value := sm.maps[index][key]
    sm.locks[index].RUnlock()
    return value
}

func (sm *SafeMap) WriteMap(key string, value string) {
    index := hash(key) % N
    sm.locks[index].Lock()
    sm.maps[index][key] = value
    sm.locks[index].Unlock()
}

func hash(s string) int {
    h := 0
    for i := 0; i < len(s); i++ {
        h = 31*h + int(s[i])
    }
    return h
}

func main() {
    safeMap := NewSafeMap()

    var wg sync.WaitGroup

    // 啟動多個goroutine進行寫操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            safeMap.WriteMap(fmt.Sprintf("name%d", i), fmt.Sprintf("John%d", i))
        }(i)
    }

    wg.Wait()

    // 啟動多個goroutine進行讀操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            fmt.Println(safeMap.ReadMap(fmt.Sprintf("name%d", i)))
        }(i)
    }

    wg.Wait()
}

在這個示例中,我們定義了一個 SafeMap 結(jié)構(gòu)體,它包含一個長度為 N 的 map 數(shù)組和一個長度為 N 的鎖數(shù)組。

定義了兩個方法:ReadMap 和 WriteMap。在這兩個方法中,我們都使用了一個 hash 函數(shù)來計算 key 應(yīng)該存儲在哪個 map 中。然后再對這個 map 進行讀寫操作。

在 main 函數(shù)中,我們啟動了多個 goroutine 來進行讀寫操作,這些操作都是安全的。

有一個開源項目 orcaman/concurrent-map 就是通過這種思想來做的,感興趣的同學(xué)可以看看。

sync.Map

最后,在內(nèi)置的 sync 包中(Go 1.9+)也有一個線程安全的 map,通過將讀寫分離的方式實現(xiàn)了某些特定場景下的性能提升。

package main

import (
    "fmt"
    "sync"
)

func main() {
    var m sync.Map
    var wg sync.WaitGroup

    // 啟動多個goroutine進行寫操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            m.Store(fmt.Sprintf("name%d", i), fmt.Sprintf("John%d", i))
        }(i)
    }

    wg.Wait()

    // 啟動多個goroutine進行讀操作
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            v, _ := m.Load(fmt.Sprintf("name%d", i))
            fmt.Println(v.(string))
        }(i)
    }

    wg.Wait()
}

有了官方的支持,代碼瞬間少了很多,使用起來方便多了。

在這個示例中,我們使用了內(nèi)置的 sync.Map 類型來存儲鍵值對,使用 Store 方法來存儲鍵值對,使用 Load 方法來獲取鍵值對。

在 main 函數(shù)中,我們啟動了多個 goroutine 來進行讀寫操作,這些操作都是安全的。

總結(jié)

Go 語言中的 map 本身并不是并發(fā)安全的。

在多個 goroutine 同時訪問同一個 map 時,可能會出現(xiàn)并發(fā)不安全的現(xiàn)象。這是因為 Go 語言中的 map 并沒有內(nèi)置鎖來保護對map的訪問。

盡管如此,我們?nèi)匀豢梢允褂靡恍┓椒▉韺崿F(xiàn) map 的并發(fā)安全。

一種方法是使用讀寫鎖,在讀操作時加讀鎖,在寫操作時加寫鎖。

另一種方法是分片加鎖,將這個 map 分成 n 塊,每個塊之間的讀寫操作都互不干擾,從而降低沖突的可能性。

此外,在內(nèi)置的 sync 包中(Go 1.9+)也有一個線程安全的 map,它通過將讀寫分離的方式實現(xiàn)了某些特定場景下的性能提升。

以上就是本文的全部內(nèi)容,如果覺得還不錯的話歡迎點贊,轉(zhuǎn)發(fā)和關(guān)注,感謝支持。

參考文章:

  • https://zhuanlan.zhihu.com/p/356739568

責(zé)任編輯:武曉燕 來源: AlwaysBeta
相關(guān)推薦

2024-04-07 00:04:00

Go語言Map

2022-11-22 08:01:30

2022-01-10 23:54:56

GoMap并發(fā)

2022-04-06 08:19:13

Go語言切片

2024-01-01 08:10:40

Go語言map

2024-01-05 08:45:35

Go語言map

2021-06-08 11:15:10

Redis數(shù)據(jù)庫命令

2022-03-04 10:07:45

Go語言字節(jié)池

2021-07-30 07:28:15

WorkerPoolGo語言

2023-12-21 07:09:32

Go語言任務(wù)

2013-05-28 09:43:38

GoGo語言并發(fā)模式

2021-07-15 23:18:48

Go語言并發(fā)

2023-11-30 08:09:02

Go語言

2023-05-19 08:01:57

Go 語言map

2023-07-28 08:04:56

StringHeaatomic線程

2023-11-21 15:46:13

Go內(nèi)存泄漏

2012-06-15 09:56:40

2023-02-10 09:40:36

Go語言并發(fā)

2019-05-15 11:38:22

GoogleGo編程語言

2024-12-31 11:40:05

點贊
收藏

51CTO技術(shù)棧公眾號