在 Go 语言中,构造一个并发安全的 map 集合
Map 集合是 Go 中提供的一个 KV 结构的数据类型,对它的操作在实际的开发中应该是非常多的,不过它不是一个线程安全的。
1 、Map 不是线程安全的
编写下面的测试代码:
func TestUnsafeMap(t *testing.T) {
// 创建一个线程不安全的map
myMap := make(map[int]int)
// 创建一个WaitGroup用于等待所有goroutine完成
var wg sync.WaitGroup
// 启动多个goroutine并发访问map
for i := 0; i < 5; i++ {
wg.Add(1)
go func(i int) {
defer wg.Done()
myMap[i] = i * 10 // 向map中写入数据
fmt.Printf("Key: %d, Value: %d\n", i, myMap[i]) // 读取map中的数据
}(i)
}
// 等待所有goroutine完成
wg.Wait()
// 检查map中的值是否正确(这一步是为了模拟测试)
for i := 0; i < 5; i++ {
if myMap[i] != i*10 {
t.Errorf("Expected %d but got %d", i*10, myMap[i])
}
}
}
在上面的代码中,虽然只是循环了很少的次数,就很容易出现竞争问题,错误信息类似下面的:
=== RUN TestUnsafeMap
Key: 4, Value: 40
Fatal error: concurrent map writesKey: 0, Value: 0
Key: 2, Value: 20
Key: 1, Value: 10 t
2 、构建线程安全的 Map
在 Go 中,为了实现一个线程安全的 map 操作,可以使用 sync.Map 或者通过互斥锁(sync.Mutex)来实现。下面的示例代码是使用锁来手动实现的:
type SaleMap struct {
mu sync.Mutex
m map[int]int
}
func NewSaleMap() *SaleMap {
return &SaleMap{m: make(map[int]int)}
}
func (s *SaleMap) Set(key int, value int) {
s.mu.Lock()
defer s.mu.Unlock()
s.m[key] = value
}
func (s *SaleMap) Get(key int) (int, bool) {
s.mu.Lock()
defer s.mu.Unlock()
value, ok := s.m[key]
return value, ok
}
然后编写单测如下:
func TestConcurrentMap(t *testing.T) {
// 创建一个线程安全的map
safeMap := NewSaleMap()
var wg sync.WaitGroup
// 启动多个goroutine并发写入和读取数据
for i := 0; i < 500000; i++ {
wg.Add(1)
go func(i int) {
defer wg.Done()
safeMap.Set(i, i*10) // 向map中写入数据
}(i)
}
// 等待所有goroutine完成
wg.Wait()
// 检查map中的值是否正确(这一步是为了模拟测试)
for i := 0; i < 500000; i++ {
if value, _ := safeMap.Get(i); value != i*10 {
t.Errorf("Expected %d but got %d", i*10, value)
}
}
}
这一次,循环了更多次,不会再有并发问题了。
Over!
在 Go 语言中,构造一个并发安全的 map 集合的更多相关文章
- 在 Go 语言中,我为什么使用接口
强调一下是我个人的见解以及接口在 Go 语言中的意义. 如果您写代码已经有了一段时间,我可能不需要过多解释接口所带来的好处,但是在深入探讨 Go 语言中的接口前,我想花一两分钟先来简单介绍一下接口. ...
- [原创]C/C++语言中,如何在main.c或main.cpp中调用另一个.c文件
C/C++语言中,如何在main.cpp中调用另一个.c文件主要有5种思路: 1.在VS2012 IDE中,将被引用的.c文件后缀名全部修改为.h,然后通过IDE的解决方案资源管理器中鼠标右键单击“头 ...
- Go 语言中,有时 nil 并不是一个 nil
今天,我遇到了一个 Go FAQ.首先,作为一个小小的 Go 语言测验,看看您是否在 Go playground 中运行该程序之前就能推断出它应该打印出的内容(我已经将程序放在侧边栏中,以防它在 Go ...
- Java 语言中 Enum 类型的使用介绍
Enum 类型的介绍 枚举类型(Enumerated Type) 很早就出现在编程语言中,它被用来将一组类似的值包含到一种类型当中.而这种枚举类型的名称则会被定义成独一无二的类型描述符,在这一点上和常 ...
- 【翻译】go语言中的map实战
业余时间翻译,水平很差,如有瑕疵,纯属无能. 原文链接 http://blog.golang.org/go-maps-in-action go语言中的map实战 1. 简介 哈希表是计算机科学中最重要 ...
- C语言中setjmp与longjmp学习笔记
C语言中setjmp与longjmp学习笔记 一.基础介绍 头文件:#include<setjmp.h> 原型: int setjmp(jmp_buf envbuf) ,然而longjm ...
- 浅析C语言中assert的用法(转)
原文地址:http://www.jb51.net/article/39685.htm 以下是对C语言中assert的使用方法进行了介绍,需要的朋友可以参考下. assert宏的原型定义在<ass ...
- C语言中,头文件和源文件的关系(转)
简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程: 1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句, ...
- C 语言中 setjmp 和 longjmp
在 C 语言中,我们不能使用 goto 语句来跳转到另一个函数中的某个 label 处:但提供了两个函数——setjmp 和 longjmp来完成这种类型的分支跳转.后面我们会看到这两个函数在处理异常 ...
- c语言中的scanf在java中应该怎么表达,Scanner类。
1 java是面向对象的语言 它没有像C语言中的scanf()函数,但是它的类库中有含有scanf功能的函数 2 java.util包下有Scanner类 Scanner类的功能与scanf类似 3 ...
随机推荐
- Vue项目实战:构建你的第一个项目
Vue项目实战:从零到一构建你的第一个应用 准备工作 在开始使用Vue之前,请确保您已经安装了Node.js 16.0或更高版本.Node.js是运行Vue项目所必需的JavaScript运行环境. ...
- 硬件设计:POE--POE基础
参考资料:POE供电基础知识:PSE PD检测过程详解 POE供电简介 以太网供电 一.POE相关介绍 POE(Power Over Ethernet)是指在现有的以太网Cat.5布线基础架构上不做任 ...
- initDB.sh初始化磁盘脚本centos7
新加磁盘初始化脚本 跳转:优化(2022-4-14) vim initDB.sh #!/bin/bash # auther by wangxp EXCLUDE_LIST='2,11' EXCLUDE_ ...
- Integer超过128要用对象比较,否则出问题
一.测试代码 public void testEquals() { int int1 = 12; int int2 = 12; Integer integer1 = new Integer(12); ...
- ORACLE SELECT INTO 赋值为空,抛出 NO DATA FOUND 异常
例子: DECLARE ORDER_NUM VARCHAR2(20); BEGIN SELECT S.ORDER_NUM INTO ORDER_NUM FROM SALES_ORDER S WHERE ...
- 一种将历史地图坐标配准到GIS中的方法
经常我们看到历史地图影像,比如谭图里面的各个历史朝代的大地图, 然后我们希望利用这个影像作为图层或者叫底图,然后在GIS软件上编辑一些矢量文件, 从而产生的地图矢量文件具有真实的经纬度坐标,不是单单的 ...
- redis - [02] 安装部署
在Windows和Linux操作系统下的安装部署 一.windows版 (1)访问redis下载地址:https://github.com/tporadowski/redis/releases (2) ...
- Linux - 服务器磁盘 Raid & 分区 & 挂载
一.流计算服务器 有一台流处理服务器(系统盘:2*600G.数据盘:6*600G)分区挂载如下: 设备名 分区 大小 挂载点 文件系统类型 磁盘用途 分区类别 /dev/sda /dev/sda1 3 ...
- VulnHub-DC-6靶机-wpscan爆破+命令注入反弹shell+nmap提权
一.环境搭建 选择扫描虚拟机 选择靶机路径 这里如果出现报错,无法导入,如VMware出现配置文件 .vmx 是由VMware产品创建,但该产品与此版 VMware workstation 不兼容,因 ...
- php对接股票、期货数据源API接口
以下是使用 PHP 对接 StockTV API 的项目实现.我们将使用 cURL 进行 HTTP 请求,并使用 Ratchet 处理 WebSocket 连接. 项目结构 stocktv-api-p ...