基本同步原语
Mutex
- 互斥锁 Mutual exclusion
- 任何时间只允许一个 goroutine 在临界区运行
- 避免死锁
- 公平
- 零值是未锁状态
- Unlock 未加锁的 Mutex 会 panic
- 加锁的 Mutex 不和这个特定的 goroutine 关联
- 非重入锁
RWMutex
- 可以被一堆的reader持有,或者被一个 writer 持有
- 适合大并发 read 的场景
- 零值是未加锁状态
- writer 的 Lock相对后续 reader 的 RLock 优先级高
- 禁止递归读锁(防止死锁)
Cond (不常用)
- Mutex有些不适用
- Monitor vs Mutex. Monitor = Mutex + Condition Variables
- Condition variable 是一组等待同意一个条件的goroutine的容器
- 每个 cond 和一个Locker 相关联
- 改变条件或者调用 Wait 需要获取锁
WaitGroup
- 等待一组goroutine完成(java CountdownLatch/CyclicBarrier)
- Add 参数可以是负数,如果计数器小于0,panic
- 当计数器为0的时候,阻塞在 Wait 方法的 goroutine 都会被释放
- 可重用, but...
- 注意不要把Add方法放到 goroutine 里和wait 一起执行
Once
- 只执行一次初始化 func (o *Once) Do(f func())
- 避免死锁
- 即使 f panic, Once 也认为它完成了
单例 - 常量 - package 变量(eager) - init 函数 (eager) - GetInstance()(lazy) - 通过 sync.Once 或者类似实现
Pool
- 临时对象池
- 可能在任何时候任意对象都可能被移除
- 可以安全并发访问
- 装箱/拆箱
Sync.Map
- 适用于两个 case(空间换时间):
- 设置一次,多次读(cache)
- 多个 goroutine 并发的读,写,更新不同的 key
- 装箱/拆箱
- Range 进行遍历,可能会加锁
- 没有 Len 方法,并且也不会添加
扩展同步原语
ReentrantLock
type RecursiveMutext struct {
sync.Mutex
owner int64
recursion int32
}
Semaphore
- Dijkstra 提出并发访问通用资源的并发原语 (sync/semaphore)
- 初始化一个非负的值 S
- P(wait)减1,如果 S 小于0 ,阻塞本 goroutine 进入临界区
- V(signal)加1, 如果 S 不为负值, 其他goroutine可以进入临界区
- 二进制信号量可以实现锁(0,1)
- 计数信号量
实现: sync/semaphore sync/singleflight
ErrGroup
- Wait会等待所有 goroutine 执行完才释放
- 如果想遇到第一个 err 就返回,用 Context
SpinLock (自旋锁)
- 有时候很高效
- 公平性
- 处理器忙等待
fslock 文件锁
- github.com/juju/fslock
concurrent-map
- github.com/orcaman/concurrent-map
原子操作
atomic 数据类型
- int32, int64, uint32, uint64, uintptr, unsafe.Pointer
- AddXXX(整数类型)
- CompareAndSwapXXX: cas
- LoadXXX: 读取
- StoreXXX: 存储
- SwapXXX: 交换
有Add没有 Subtract?
- 有符号类型,可以用 Add 负数
- 无符号类型,可以用 AddUint32(&x, ^uint32(c-1)), AddUint64(&x, ^uint64(c-1))
- 无符号类型减一,AddUint32(&x, ^uint32(0)), AddUint64(&x, ^uint64(0))
atomic.value
复杂类型进行原子存储和加载
- Load
- Store
Channel
- 信号(shutdown/close/finish)
- 数据交流(queue/stream)
- 锁(mutex)
Channel vs Mutex
过度使用channel 和goroutine
- Channel
- 传递数据的owner
- 分发任务单元
- 交流异步结果
- 任务编排
- Mutex
- cache
- 状态
- 临界区
OrDone
内存模型
描述线程(goroutine)通过内存交互,以及对数据的共享使用。
Go内存模型:
happend before
- goroutine的创建 happens before 所有此 goroutine 中的操作
- goroutine的销毁 happens after 所有此 goroutine 中的操作