美文网首页
golang_并发安全: slice和map并发不安全及解决方法

golang_并发安全: slice和map并发不安全及解决方法

作者: 彩色代码 | 来源:发表于2020-07-21 13:42 被阅读0次

并发安全

并发安全也叫线程安全,在并发中出现了数据的丢失,称为并发不安全

map和slice都是并发不安全的

切片并发不安全

场景: 10000个协程同时添加切片

var s []int

func appendValue(i int) {
    s = append(s, i)
}

func main() {
    for i := 0; i < 10000; i++ { //10000个协程同时添加切片
        go appendValue(i)
    }

    for i, v := range s {       //同时打印索引和值
        fmt.Println(i, ":", v)
    }
}

结果:

8194 : 9923
8195 : 9924
8196 : 9925
8197 : 9926
8198 : 9927
8199 : 9928
8200 : 9929
8201 : 9930

没有到:9999 ,说明数据有丢失

解决方法: 加锁

var s []int
var lock sync.Mutex //互斥锁
func appendValue(i int) {
    lock.Lock() //加锁
    s = append(s, i)
    lock.Unlock() //解锁
}

func main() {
    for i := 0; i < 10000; i++ {
        go appendValue(i)
    }
    //sort.Ints(s) //给切片排序,先排完序再打印,和下面一句效果相同
    time.Sleep(time.Second)  //间隔1s再打印,防止一边插入数据一边打印时数据乱序
    for i, v := range s {
        fmt.Println(i, ":", v)
    }
}

map并发不安全

场景: 2个协程同时读和写

func main() {
    m := make(map[int]int)
    go func() {             //开一个协程写map
        for i := 0; i < 10000; i++ {
            m[i] = i
        }
    }()

    go func() {             //开一个协程读map
        for i := 0; i < 10000; i++ {
            fmt.Println(m[i])
        }
    }()

    //time.Sleep(time.Second * 20)
    for {
        ;
    }
}

结果:

0
fatal error: concurrent map read and map write

解决方法:尽量不要做map的并发,如果用并发要加锁,保证map的操作要么读,要么写。

var lock sync.Mutex
func main() {
    m:=make(map[int]int)
    go func() {         //开一个协程写map
        for i:=0;i<10000 ;i++  {
            lock.Lock()             //加锁
            m[i]=i
            lock.Unlock()           //解锁
        }
    }()
    go func() {         //开一个协程读map
        for i:=0;i<10000 ;i++  {
            lock.Lock()             //加锁
            fmt.Println(m[i])
            lock.Unlock()           //解锁
        }
    }()
    time.Sleep(time.Second*20)
}

相关文章

  • golang_并发安全: slice和map并发不安全及解决方法

    并发安全 并发安全也叫线程安全,在并发中出现了数据的丢失,称为并发不安全 map和slice都是并发不安全的 切片...

  • Go sync.Map

    map并发读线程安全,并发读写线程不安全。 sync.Map 读写分离 空间换时间 Map Golang1.6之前...

  • Go Map 为什么是非线程安全的?

    Go map 默认是并发不安全的,同时对 map 进行并发读写的时,程序会 panic,原因如下:Go 官方经过长...

  • know un known

    1 线程安全与不安全的区别 2 并发与并行,c#的高并发解决方案有哪些 3 并发还分为io并发和cpu并发 4 长...

  • golang并发map记录

    1.普通map+mutex,实现map的并发安全读写 2.sync.Map 3.第三方并发安全map库

  • 4、ConcurrentHashMap实现原理

    1. 各种map线程安全介绍 1.1 HashMap HashMap是线程不安全的,在并发环境下,可能会形成环状链...

  • JAVA-ConcurrentHashMap源码解析

    ConcurrentHashMap是一个兼具高并发和并发安全的一个 Map 类,主要依赖于 Unsafe 类,闲话...

  • Go语言高并发Map解决方案

    Go语言高并发Map解决方案 Go语言基础库中的map不是并发安全的,不过基于读写锁可以实现线程安全;不过在Go1...

  • 并发安全的有锁map

    并发安全的有锁map 直接上代码吧:

  • sync.Map && map

    面试题: 为什么map不能并发读写? map 并发读写会panic吗? map + lock 和 sync.Ma...

网友评论

      本文标题:golang_并发安全: slice和map并发不安全及解决方法

      本文链接:https://www.haomeiwen.com/subject/asjikktx.html