go并发访问map的坑 fatal error: concurrent map read and map write

前言

go 并发访问map是不安全的, 会出现未定义的错误导致程序退出…坑总得踩一踩才会知道

例子

先写一个错误的

package main

import "fmt"

func test() map[string]interface{}{
	taskList := []string{"a", "b"}
	result := make(map[string]interface{})
	for _, task := range taskList {
		go func(task string) {
			switch task {
			case "a":
				result["a"] = "xiaofei"
			case "b":
				result["b"] = "ceshi"
			}
		}(task)
	}
	return result
}
func main() {
	for a := 0; a < 1000; a++ {
		fmt.Println(test())
	}
}

返回结果
在这里插入图片描述
上面代码有两个问题, 第一个是没有阻塞go程, 第二个就是并发访问map了

正确的写法

package main

import (
	"fmt"
	"sync"
)

// 使用读写锁 sync.RWMutex
type NewMap struct {
	lock *sync.RWMutex
	sm   map[interface{}]interface{}
}

func (m *NewMap) Set(k interface{}, v interface{}) bool {
	m.lock.Lock()
	defer m.lock.Unlock()
	m.sm[k] = v
	return true
}
func test() map[interface{}]interface{} {
	taskList := []string{"a", "b"}
	xf := &sync.WaitGroup{}
	result := NewMap{
		lock: new(sync.RWMutex),
		sm: make(map[interface{}]interface{}),
	}
	for _, task := range taskList {
		xf.Add(1)
		go func(task string) {
			defer xf.Done()
			switch task {
			case "a":
				result.Set("a", "xiaofei")
			case "b":
				result.Set("b", "ceshi")
			}
		}(task)
	}
	xf.Wait()
	return result.sm
}
func main() {
	for a := 0; a < 100; a++ {
		fmt.Println(test())
	}
}

建议把这个map做一个基类方法, 所有的方法封装一下直接用就ok了;

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小僵尸打字员

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值