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

Grayan · · 3430 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

并发安全

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

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在并发执行中会直接报错


有疑问加站长微信联系(非本文作者)

本文来自:CSDN博客

感谢作者:Grayan

查看原文:golang_并发安全: slice和map并发不安全及解决方法

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

3430 次点击  
加入收藏 微博
2 回复  |  直到 2022-02-09 17:22:21
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传