当前位置: 首页 > news >正文

Golang 线程安全与 sync.Map

前言

线程安全通常是指在并发环境下,共享资源的访问被适当地管理,以防止竞争条件(race conditions)导致的数据不一致
Go语言中的线程安全可以通过多种方式实现

实现方式

  1. 互斥锁(Mutexes)
    Go的sync包提供了Mutex和RWMutex类型来确保在一个时间点只有一个协程可以访问某个资源
import "sync"var mu sync.Mutex
var sharedResource map[string]intfunc updateResource(key string, value int) {mu.Lock()         // 加锁sharedResource[key] = valuemu.Unlock()       // 解锁
}
  1. 原子操作(Atomic operations)
    sync/atomic包提供了一系列原子操作函数,可用于管理基本数据类型的并发访问
import "sync/atomic"var count int64func increment() {atomic.AddInt64(&count, 1) // 原子地增加计数
}
  1. 通道(Channels)
    通过使用通道,可以在协程之间安全地传递数据。当数据通过通道从一个协程传递到另一个协程时,不需要额外的同步机制
ch := make(chan int)// 发送者
go func() {ch <- 42
}()// 接收者
go func() {value := <-chfmt.Println(value)
}()
  1. 不可变性(Immutability)
    不修改数据可以自然地避免并发问题。设计数据结构和算法时,尽可能使数据不可变,可以减少同步的需要

  2. 其他同步原语
    sync包还提供了其他同步原语,如WaitGroup、Once、Cond等,可以用来同步协程的不同行为

使用上述任何一种机制时,都需要仔细设计代码以避免死锁、活锁或饥饿等问题。在Go中,可以使用go run -race命令来检测代码中的竞争条件

sync.Map

sync.Map 是一个线程安全的映射(map),它是在 sync 包中提供的。与使用互斥锁来保护普通的 map 不同,sync.Map 使用了一种无锁的技术,特别适用于以下两种场景:

  1. 当给定键的条目只写入一次但读取多次时,比如在全局缓存中
  2. 当多个协程读取、写入和覆盖不相交的键集的条目时

sync.Map 提供了一些内置方法来操作线程安全的键值对:

  • Store(key, value): 存储键值对
  • Load(key): 根据键获取值
  • LoadOrStore(key, value): 获取或存储键值对。如果键已经存在,则返回现有的键值对和 false;如果不存在,则存储并返回键值对和 true
  • Delete(key): 删除键值对
  • Range(f func(key, value interface{}) bool): 遍历所有键值对,对每个键值对执行给定的函数 f

看下基本用法

import ("fmt""sync"
)func main() {var sm sync.Map// 存储键值对sm.Store("hello", "world")sm.Store(1, 3)// 读取键对应的值if value, ok := sm.Load("hello"); ok {fmt.Println("hello:", value)}// 删除键sm.Delete("hello")// 遍历所有键值对sm.Range(func(key, value interface{}) bool {fmt.Println(key, value)return true // 继续遍历})
}

请注意,尽管 sync.Map 提供了线程安全的操作,但是它的性能通常会比使用互斥锁保护的普通 map 差,因此只推荐在上述特定场景中使用

http://www.lryc.cn/news/276321.html

相关文章:

  • 1.2 Hadoop概述
  • Adams许可管理安全控制策略
  • 无人地磅系统|内蒙古中兴首创无人地磅和远程高效管理的突破
  • 【SpringCloud】7、Spring Cloud Gateway限流配置
  • 【gRPC学习】使用go学习gRPC
  • C语言中常用的字符串函数(strlen、sizeof、sscanf、sprintf、strcpy)
  • 域名解析服务器:连接你与互联网的桥梁
  • 理论物理在天线设计和射频电路设计中的应用
  • MySql01:初识
  • Python——运算符
  • 赋能软件开发:生成式AI在优化编程工作流中的应用与前景
  • 通过盲对抗性扰动实时击败基于DNN的流量分析系统
  • 【Project】TPC-Online Module (manuscript_2024-01-07)
  • 通过cpolar在公网访问本地网站
  • Prokka: ubuntu安装的时候出现错误
  • 安全防御之密码技术
  • 一文读懂「多模态大模型」
  • 深入PostgreSQL:高级函数用法探索
  • huggingface实战bert-base-chinese模型(训练+预测)
  • CCS安装和导入项目及编译教程
  • 在React里面使用mobx状态管理详细步骤
  • 1.6PTA集练7-5~7-24、7-1、7-2,堆的操作,部落冲突(二分查找)
  • uniapp向上拉加载,下拉刷新
  • 目标检测脚本之mmpose json转yolo txt格式
  • 大数据技术在民生资金专项审计中的应用
  • 视觉SLAM十四讲|【四】误差Jacobian推导
  • 「实战应用」如何用DHTMLX Gantt构建类似JIRA式的项目路线图(一)
  • 【习题】应用程序框架
  • java基于ssm的线上选课系统的设计与实现论文
  • 汽车雷达:实时SAR成像的实现