zoukankan      html  css  js  c++  java
  • golang_并发安全: slice和map并发不安全及解决方法

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

    Grayan · 2020-07-21 15:32:48 · 1771 次点击 · 预计阅读时间 1 分钟 · 不到1分钟之前 开始浏览    
    这是一个创建于 2020-07-21 15:32:48 的文章,其中的信息可能已经有所发展或是发生改变。

    并发安全

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

    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)
    	}
    }
    

    Output:

    slice并发不安全

    没有到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)
    	}
    }
    

    总结: slice在并发执行中不会报错,但是数据会丢失

    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 {
    		;
    	}
    }
    

    Output:

    map并发不安全

    解决方法:尽量不要做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)
    }
    

    总结: map在并发执行中会直接报错

  • 相关阅读:
    arp -s 157.55.85.212 00-aa-00-62-c6-09 .... Adds a static entry.
    怎么查询局域网内全部电脑IP和mac地址等信息?
    服务器网页GZIP压缩怎么配置
    windows远程桌面端口修改
    防止ARP欺骗的方法!!!
    怎么查询局域网内全部电脑IP和mac地址..
    DedeTag Engine Create File False提示的种种原因及解决方法
    Native&nbsp;SQL
    冲销物料凭证:&#039;MBST&#039;&nbsp;VS&nbsp;&#039;MBRL&#039;
    BAPI_GOODSMVT_CREATE物料凭证创建…
  • 原文地址:https://www.cnblogs.com/ExMan/p/15246072.html
Copyright © 2011-2022 走看看