无锁队列的实现-循环数组(用循环数组实现队列)

网友投稿 286 2022-07-27

通过CAS操作免锁设计:

CAS原子 操作(Compare & Set):包含三个操作数,内存值V、旧的预期值 oldval、要修改的新值newval,当且仅当内存V中的值和旧值oldval相同时,将内存V修改为newval。

数组队列是一个循环数组,队列少用一个元素,当头等于尾标示队空,尾加1等于头标示队满。

数组的元素用EMPTY(无数据,标示可以入队)和FULL(有数据,标示可以出队)标记指示,数组一开始全部初始化成 EMPTY标示空队列。

EnQue 操作:如果当前队尾位置为EMPTY,标示线程可以在当前位置入队,通过CAS原子操作把该位置设置为FULL,避免其它线程操作这个位置,操作完后修改队尾位置。各个线程竞争新的队尾位置。如下图所示:

线程T1/T2竞争队尾位置。

T1竞争成功,首先设置FULL标记,然后对该位置进行操作。

T2轮询该位置标识为FULL继续轮询。

T1操作完成后将队尾位置后移。

T1/T2又开始竞争新的队尾。

DeQue 操作:如果当前队头位置为FULL,标示线程可以在当前位置出队,通过CAS原子操作把该位置设置为EMPTY,避免其它线程操作这个位置,操作完后修改队头位置。各个线程竞争新的队头位置。

操作没有加锁,每个线程都假设没有冲突的去完成操作,如果因为冲突失败就重试。

通过CAS、FAA、FAS操作免锁设计:

FAA操作:原子加1操作,返回更新前的值。

FAS操作:原子减1操作,返回更新前的值。

增加writeableCnt指示队列还可以写入元素个数,readableCnt指示队列中存在的元素个数。用来控制可以并发操作的线程个数。

EnQue 操作:通过原子加操作给每个要求操作的线程分配为唯一一个位置信息存放在局部变量pos中,各个线程并行的操作对应位置的信息,不再需要轮询等待。如下图所示:

T1/T2线程初始操作队尾的两个位置。

T1操作完后直接操作下一个队尾位置。

DeQue 操作:如果当前队头位置为FULL,标示线程可以在当前位置出队,通过CAS原子操作把该位置设置为EMPTY,避免其它线程操作这个位置,操作完后修改队头位置。各个线程竞争新的队头位置。

多个线程可以同时进行入队,避免了在同一个位置等待轮询,对效率有明显提升。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:SQLServer 2012之AlwaysOn —— 指定数据同步链路,消除网络抖动导致的提交延迟问题(sqlserver无法连接到服务器怎么办)
下一篇:Eclipse logo集成开发环境 Eclipse(eclipse乱码问题如何解决)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~