多线程模型下的无锁编程「建议收藏」

多线程模型下的无锁编程「建议收藏」多线程模式是比较流行的一种并发编程模型,多线程编程的一个特点就是线程间共享内存空间;这可以降低线程间通信的开销,但却引来了另外的一个难缠的问题:竟态条件!,因此,甚至有人对多线程模型提出了质疑,看这里。在多线程编程模型下,解决竟态条件的传统方法就是加锁保护临界区,但这存在影响系统性能、优先级反转等问题.因此又有人提出了,多线程模型下无锁编程的一些方式:1.线程内通信框架:Di

大家好,又见面了,我是你们的朋友全栈君。多线程模式是比较流行的一种并发编程模型,多线程编程的一个特点就是线程间共享内存空间;这可以降低线程间通信的开销,但却引来了另外的一个难缠的问题:竟态条件!,因此,甚至有人对多线程模型提出了质疑,看
这里





在多线程编程模型下,解决竟态条件的传统方法就是加锁保护临界区,但这存在影响系统性能、优先级反转等问题.




因此又有人提出了,多线程模型下无锁编程的一些方式:


1.线程内通信框架: Disruptor, 这是一款开源的并发框架,用于线程间无锁的共享数据,看
这里





2.无锁数据结构


无锁数据结构一般基于一个很重要的操作:CAS–Compare And Swap(看
这里
)。


用C语言表达的一个CAS实现的操作是这样的:

  1. /*定义CAS操作*/
  2. #define CAS __sync_bool_compare_and_swap
  3. /*
  4. * 定义stack的数据结构
  5. */
  6. typedef struct stack_node {

  7.     struct node *next;
  8.     void *data;
  9. }stack_node;
  10. /*栈顶指针*/
  11. stack_node *top = NULL;



CAS的一个重要特性是其必须是原子操作。现在大多数CPU都支持指令级别的CAS操作。


GCC编译也提供了这样的接口:bool __sync_bool_compare_and_swap (type *ptr, type oldval type newval, …)




有了这个原子的CAS后,我们就能实现自己的无锁数据结构了,下面我们就尝试着来实现一个无锁栈:




下面的代码中,我们假设malloc与free是线程安全的(至于malloc与free的可重入性与线程安全问题,可以看
这里

这里
)

首先定义必要的数据结构:

  1. /*定义CAS操作*/
  2. #define CAS __sync_bool_compare_and_swap
  3. /*
  4. * 定义stack的数据结构
  5. */
  6. typedef struct stack_node {

  7.     struct node *next; /*指向下一个节点,即通过链表的形式实现栈*/
  8.     void *data;/*指向该节点的数据*/
  9. }stack_node;
  10. /*栈顶指针*/
  11. stack_node *top = NULL;

接下来是
栈操作:

  1. /*压栈操作*/
  2. void stack_push(void *data)
  3. {

  4.     stack_node *new = malloc(sizeof(struct node));
  5.     new>data = data;
  6.     do {

  7.         new>next = top;/*获取top的快照, 同时初始化了new的next指针*/
  8.     }while(!CAS(&top, new>next, new));
  9. }

在取得top的快照后后, 就通过CAS操作查看top的内容与line 7取得的快照是否一致,如果一致则将top的内容更换为new,CAS函数返回true。

假设有线程A在
stack_push里获取top快照后暂时失去了执行权, 切换至另一个线程B, 而线程B完成了一次
stack_push操作,然后执行权再次回到线程A,

线程A执行CAS操作, 发现top里的内容与快照里的内容已经不一致了,CAS返回false, 估do…while语句重新执行。



接下来是出栈操作:

  1. /*出栈操作*/
  2. void * stack_pop(void)
  3. {

  4.     stack_node *tmp;
  5.     void *data;
  6.     
  7.     do {

  8.         tmp = top;
  9.         if (!tmp) return NULL;
  10.     }while(CAS(&top, tmp, tmp>next) = true);
  11.     data = tmp>data;
  12.     free(tmp);
  13.     return data;
  14. }

其免锁原理与入栈基本一样,估不再赘述。



另外还有一点, 在使用CAS实现免锁数据结构时, 容易出现ABA问题, 这里也暂时不作讨论, 可以从参考资料中得到更多的信息。




参考资料:


1.来自酷客的无锁队列


2.
设计不用互斥锁的并发数据结构

3.透过linux内核看无锁编程

转载自:http://blog.chinaunix.net/uid-25424552-id-3772253.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/141681.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 分别以递归调用和迭代的方法求数列_迭代算法和递归算法

    分别以递归调用和迭代的方法求数列_迭代算法和递归算法对于数列,递归和迭代的联系非常紧密。a0,a1,a2,…,an−1,ana_0,a_1,a_2,…,a_{n-1},a_na0​,a1​,a2​,…,an−1​,an​数列就是一串数字,数列来源于生活,有用的数列中蕴含着规则。要完整描述一个数列,方法有二:通项公式an=f(n)a_n=f(n)an​=f(n)递推公式其中通项公式是最一般的情况。由通项公式可以求得任意一…

    2022年9月4日
    3
  • window安装JAVA环境

    window安装JAVA环境java环境安装包:JDK11.0.2和Tomcat7.0.92JDK1.8.01、下载JDK下载JDK:JDK官网点击安装,然后点击下一步,知道安装完毕。注:下载的JDK注意要和自己的系统匹配,安装过程中如果不想使用默认安装路径,可以自行配置。配置环境变量:安装完成后,右击"我的电脑",点击"属性",选择"高级系统设置";选择"高级"选项卡,点击"环境变量…

    2022年7月16日
    13
  • Java设计模式(一)之创建型模式:工厂模式(简单工厂模式+工厂方法模式)

    Java设计模式(一)之创建型模式:工厂模式(简单工厂模式+工厂方法模式)

    2021年4月9日
    161
  • 如何配置tomcat环境变量

    如何配置tomcat环境变量首先下载tomcat,并且解压到目录:注意:2,3步的变量值要到下图这一步即,bin的上一级目录不包含bin1.第一步鼠标右键计算机->属性->高级系统设置,进去之后,点击环境变量,如下图所示2.第二步开始配置tomcat的环境变量,新建系统变量名CATALINA_BASE,值tomcat的安装路径,如下图所示:3.第三步新建系统变量CA…

    2022年6月4日
    26
  • sublime 激活码【2021最新】

    (sublime 激活码)本文适用于JetBrains家族所有ide,包括IntelliJidea,phpstorm,webstorm,pycharm,datagrip等。https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~M…

    2022年3月21日
    80
  • gb28181协议详解_GB28181收费吗

    gb28181协议详解_GB28181收费吗ssdp协议近似于http协议,事实上,和http协议相似得地方就是他得协议内容,当然,我们要去除他得端口和d类地址。为什么我在给其他员工或者面试得时候要他人深入一些,理解一下http协议,是因为理解了http协议,掌握ssdp也就不远了,很多人可能会问http协议有啥内容,无非就是get,post,put,delete么,还能怎么样,我经常问他们一点http协议怎么知道他结束了?虽然ssdp是udp协议,但是他依然需要\r\n来代表行结束,\r\n\r\n代表协议内容部分结束。……

    2022年10月11日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号