转自:http://blog.csdn.net/minCrazy/article/details/40791795

多线程间计数操作、共享状态或者统计相关时间次数,这些都需要在多线程之间共享变量和修改变量,如此就需要在多线程间对该变量进行互斥操作和访问。

通常遇到多线程互斥的问题,首先想到的就是加锁lock,通过加互斥锁来进行线程间互斥,但是最近有看一些开源的项目,看到有一些同步读和操作的原子操作函数——__sync_fetch_and_add系列的命令,然后自己去网上查找一番,找到一篇博文有介绍这系列函数,学习一番后记录下来。

首先,C/C++程序中count++这种操作不是原子的,一个自加操作,本质上分为3步:

  1. 从缓存取到寄存器
  2. 在寄存器内加1
  3. 再存入缓存
但是由于时序的因素,多线程操作同一个全局变量,就会出现很多问题。这就是多线程并发编程的难点,尤其随着计算机硬件技术的快速发展,多CPU多核技术更彰显出这种困难。

通常,最简单的方法就是加锁保护,互斥锁(mutex),这也是我使用最多的解决方案。大致代码如下:
pthread_mutex_t lock;
pthread_mutex_init(&lock,...);

pthread_mutex_lock(&lock);
count++;
pthread_mutex_unlock(&lock);

后来,在一些C/C++开源项目中,看到通过__sync_fetch_and_add一系列命令进行原子性操作,随后就在网上查阅相关资料,发现有很多博客都有介绍这系列函数。

__sync_fetch_and_add系列一共有12个函数,分别:加/减/与/或/异或等原子性操作函数,__sync_fetch_and_add,顾名思义,先fetch,返回自加前的值。举例说明,count = 4,调用__sync_fetch_and_add(&count, 1)之后,返回值是4,但是count变成5。同样,也有__sync_add_and_fetch,先自加,然后返回自加后的值。这样对应的关系,与i++和++i的关系是一样的。

gcc从4.1.2开始提供了__sync_*系列的build-in函数,用于提供加减和逻辑运算的原子操作,其声明如下:

type __sync_fetch_and_add (type *ptr, type value, ...)
type __sync_fetch_and_sub (type *ptr, type value, ...)
type __sync_fetch_and_or (type *ptr, type value, ...)
type __sync_fetch_and_and (type *ptr, type value, ...)
type __sync_fetch_and_xor (type *ptr, type value, ...)
type __sync_fetch_and_nand (type *ptr, type value, ...)

type __sync_add_and_fetch (type *ptr, type value, ...)
type __sync_sub_and_fetch (type *ptr, type value, ...)
type __sync_or_and_fetch (type *ptr, type value, ...)
type __sync_and_and_fetch (type *ptr, type value, ...)
type __sync_xor_and_fetch (type *ptr, type value, ...)
type __sync_nand_and_fetch (type *ptr, type value, ...)

上述12个函数即为所有,通过函数名字就可以知道函数的作用。需要注意的是,这个type不能乱用(type只能是int, long, long long以及对应的unsigned类型),同时在用gcc编译的时候要加上选项 -march=i686。
后面的可扩展参数(...)用来指出哪些变量需要memory barrier,因为目前gcc实现的是full barrier(类似Linux kernel中的mb(),表示这个操作之前的所有内存操作不会被重排到这个操作之后),所以可以忽略掉这个参数。

下面简单介绍一下__sync_fetch_and_add反汇编出来的指令(实际上,这部分我还不是很懂,都是从其他博客上摘录的)
804889d:f0 83 05 50 a0 04 08 lock addl $0x1,0x804a050
可以看到,addl前面有一个lock,这行汇编指令前面是f0开头,f0叫做指令前缀,Richard Blum。lock前缀的意思是对内存区域的排他性访问。

其实,lock是锁FSB,前端串行总线,Front Serial Bus,这个FSB是处理器和RAM之间的总线,锁住FSB,就能阻止其他处理器或者Core从RAM获取数据。当然这种操作开销相当大,只能操作小的内存可以这样做,想想我们有memcpy,如果操作一大片内存,锁内存,那么代价太大了。所以前面介绍__sync_fetch_and_add等函数,type只能是int, long, long long以及对应的unsigned类型。

此外,还有两个类似的原子操作,
bool __sync_bool_compare_and_swap(type *ptr, type oldval, type newval, ...)
type __sync_val_compare_and_swap(type *ptr, type oldval, type newval, ...)

这两个函数提供原子的比较和交换,如果*ptr == oldval,就将newval写入*ptr,
第一个函数在相等并写入的情况下返回true;
第二个函数在返回操作之前的值。
 
type __sync_lock_test_and_set(type *ptr, type value, ...)
将*ptr设为value并返回*ptr操作之前的值;
void __sync_lock_release(type *ptr, ...)
将*ptr置为0
 
有了这些宝贝函数,对于多线程对全局变量进行操作(自加、自减等)问题,我们就不用考虑线程锁,可以考虑使用上述函数代替,和使用pthread_mutex保护的作用是一样的,线程安全且性能上完爆线程锁。
 
下面是对线程锁和原子操作使用对比,并且进行性能测试与对比。代码来自于文献【1】,弄懂后并稍微改动一点点。代码中分别给出加锁、加线程锁、原子计数操作三种情况的比较。
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <errno.h>
  5. #include <pthread.h>
  6. #include <sched.h>
  7. #include <linux/unistd.h>
  8. #include <sys/syscall.h>
  9. #include <linux/types.h>
  10. #include <time.h>
  11. #include <sys/time.h>
  12. #define INC_TO 1000000 // one million
  13. __u64 rdtsc ()
  14. {
  15. __u32 lo, hi;
  16. __asm__ __volatile__
  17. (
  18. "rdtsc":"=a"(lo),"=d"(hi)
  19. );
  20. return (__u64)hi << 32 | lo;
  21. }
  22. int global_int = 0;
  23. pthread_mutex_t count_lock = PTHREAD_MUTEX_INITIALIZER;//初始化互斥锁
  24. pid_t gettid ()
  25. {
  26. return syscall(__NR_gettid);
  27. }
  28. void * thread_routine1 (void *arg)
  29. {
  30. int i;
  31. int proc_num = (int)(long)arg;
  32. __u64 begin, end;
  33. struct timeval tv_begin, tv_end;
  34. __u64 time_interval;
  35. cpu_set_t set;
  36. CPU_ZERO(&set);
  37. CPU_SET(proc_num, &set);
  38. if (sched_setaffinity(gettid(), sizeof(cpu_set_t), &set))
  39. {
  40. fprintf(stderr, "failed to set affinity\n");
  41. return NULL;
  42. }
  43. begin = rdtsc();
  44. gettimeofday(&tv_begin, NULL);
  45. for (i = 0; i < INC_TO; i++)
  46. {
  47. __sync_fetch_and_add(&global_int, 1);
  48. }
  49. gettimeofday(&tv_end, NULL);
  50. end = rdtsc();
  51. time_interval = (tv_end.tv_sec - tv_begin.tv_sec) * 1000000 + (tv_end.tv_usec - tv_begin.tv_usec);
  52. fprintf(stderr, "proc_num : %d, __sync_fetch_and_add cost %llu CPU cycle, cost %llu us\n", proc_num, end - begin, time_interval);
  53. return NULL;
  54. }
  55. void *thread_routine2(void *arg)
  56. {
  57. int i;
  58. int proc_num = (int)(long)arg;
  59. __u64 begin, end;
  60. struct timeval tv_begin, tv_end;
  61. __u64 time_interval;
  62. cpu_set_t set;
  63. CPU_ZERO(&set);
  64. CPU_SET(proc_num, &set);
  65. if (sched_setaffinity(gettid(), sizeof(cpu_set_t), &set))
  66. {
  67. fprintf(stderr, "failed to set affinity\n");
  68. return NULL;
  69. }
  70. begin = rdtsc();
  71. gettimeofday(&tv_begin, NULL);
  72. for (i = 0; i < INC_TO; i++)
  73. {
  74. pthread_mutex_lock(&count_lock);
  75. global_int++;
  76. pthread_mutex_unlock(&count_lock);
  77. }
  78. gettimeofday(&tv_end, NULL);
  79. end = rdtsc();
  80. time_interval = (tv_end.tv_sec - tv_begin.tv_sec) * 1000000 + (tv_end.tv_usec - tv_begin.tv_usec);
  81. fprintf(stderr, "proc_num : %d, pthread_mutex_lock cost %llu CPU cycle, cost %llu us\n", proc_num, end - begin, time_interval);
  82. return NULL;
  83. }
  84. void *thread_routine3(void *arg)
  85. {
  86. int i;
  87. int proc_num = (int)(long)arg;
  88. __u64 begin, end;
  89. struct timeval tv_begin, tv_end;
  90. __u64 time_interval;
  91. cpu_set_t set;
  92. CPU_ZERO(&set);
  93. CPU_SET(proc_num, &set);
  94. if (sched_setaffinity(gettid(), sizeof(cpu_set_t), &set))
  95. {
  96. fprintf(stderr, "failed to set affinity\n");
  97. return NULL;
  98. }
  99. begin = rdtsc();
  100. gettimeofday(&tv_begin, NULL);
  101. for (i = 0; i < INC_TO; i++)
  102. {
  103. global_int++;
  104. }
  105. gettimeofday(&tv_end, NULL);
  106. end = rdtsc();
  107. time_interval = (tv_end.tv_sec - tv_begin.tv_sec) * 1000000 + (tv_end.tv_usec - tv_begin.tv_usec);
  108. fprintf(stderr, "proc_num : %d, no lock cost %llu CPU cycle, cost %llu us\n", proc_num, end - begin, time_interval);
  109. return NULL;
  110. }
  111. int main()
  112. {
  113. int procs = 0;
  114. int all_cores = 0;
  115. int i;
  116. pthread_t *thrs;
  117. procs = (int)sysconf(_SC_NPROCESSORS_ONLN);
  118. if (procs < 0)
  119. {
  120. fprintf(stderr, "failed to fetch available CPUs(Cores)\n");
  121. return -1;
  122. }
  123. all_cores = (int)sysconf(_SC_NPROCESSORS_CONF);
  124. if (all_cores < 0)
  125. {
  126. fprintf(stderr, "failed to fetch system configure CPUs(Cores)\n");
  127. return -1;
  128. }
  129. printf("system configure CPUs(Cores): %d\n", all_cores);
  130. printf("system available CPUs(Cores): %d\n", procs);
  131. thrs = (pthread_t *)malloc(sizeof(pthread_t) * procs);
  132. if (thrs == NULL)
  133. {
  134. fprintf(stderr, "failed to malloc pthread array\n");
  135. return -1;
  136. }
  137. printf("starting %d threads...\n", procs);
  138. for (i = 0; i < procs; i++)
  139. {
  140. if (pthread_create(&thrs[i], NULL, thread_routine1, (void *)(long) i))
  141. {
  142. fprintf(stderr, "failed to pthread create\n");
  143. procs = i;
  144. break;
  145. }
  146. }
  147. for (i = 0; i < procs; i++)
  148. {
  149. pthread_join(thrs[i], NULL);
  150. }
  151. printf("after doing all the math, global_int value is: %d\n", global_int);
  152. printf("expected value is: %d\n", INC_TO * procs);
  153. free (thrs);
  154. return 0;
  155. }
 
         运行结果如下:
         每次修改不同thread_routine?()函数,重新编译即可测试不同情况。
         g++ main.cpp -D _GNU_SOURCE -l pthread
         ./a.out
 
         不加锁下运行结果:
  1. system configure CPUs(Cores): 8
  2. system available CPUs(Cores): 8
  3. starting 8 threads...
  4. proc_num : 5, no lock cost 158839371 CPU cycle, cost 66253 us
  5. proc_num : 6, no lock cost 163866879 CPU cycle, cost 68351 us
  6. proc_num : 2, no lock cost 173866203 CPU cycle, cost 72521 us
  7. proc_num : 7, no lock cost 181006344 CPU cycle, cost 75500 us
  8. proc_num : 1, no lock cost 186387174 CPU cycle, cost 77728 us
  9. proc_num : 0, no lock cost 186698304 CPU cycle, cost 77874 us
  10. proc_num : 3, no lock cost 196089462 CPU cycle, cost 81790 us
  11. proc_num : 4, no lock cost 200366793 CPU cycle, cost 83576 us
  12. after doing all the math, global_int value is: 1743884
  13. expected value is: 8000000

线程锁下运行结果:

  1. system configure CPUs(Cores): 8
  2. system available CPUs(Cores): 8
  3. starting 8 threads...
  4. proc_num : 1, pthread_mutex_lock cost 9752929875 CPU cycle, cost 4068121 us
  5. proc_num : 5, pthread_mutex_lock cost 10038570354 CPU cycle, cost 4187272 us
  6. proc_num : 7, pthread_mutex_lock cost 10041209091 CPU cycle, cost 4188374 us
  7. proc_num : 0, pthread_mutex_lock cost 10044102546 CPU cycle, cost 4189546 us
  8. proc_num : 6, pthread_mutex_lock cost 10113533973 CPU cycle, cost 4218541 us
  9. proc_num : 4, pthread_mutex_lock cost 10117540197 CPU cycle, cost 4220212 us
  10. proc_num : 3, pthread_mutex_lock cost 10160384391 CPU cycle, cost 4238083 us
  11. proc_num : 2, pthread_mutex_lock cost 10164464784 CPU cycle, cost 4239778 us
  12. after doing all the math, global_int value is: 8000000
  13. expected value is: 8000000

原子操作__sync_fetch_and_add下运行结果:

  1. system configure CPUs(Cores): 8
  2. system available CPUs(Cores): 8
  3. starting 8 threads...
  4. proc_num : 3, __sync_fetch_and_add cost 2364148575 CPU cycle, cost 986129 us
  5. proc_num : 1, __sync_fetch_and_add cost 2374990974 CPU cycle, cost 990652 us
  6. proc_num : 2, __sync_fetch_and_add cost 2457930267 CPU cycle, cost 1025247 us
  7. proc_num : 5, __sync_fetch_and_add cost 2463027030 CPU cycle, cost 1027373 us
  8. proc_num : 7, __sync_fetch_and_add cost 2532240981 CPU cycle, cost 1056244 us
  9. proc_num : 4, __sync_fetch_and_add cost 2555055054 CPU cycle, cost 1065760 us
  10. proc_num : 0, __sync_fetch_and_add cost 2561248971 CPU cycle, cost 1068331 us
  11. proc_num : 6, __sync_fetch_and_add cost 2558781396 CPU cycle, cost 1067314 us
  12. after doing all the math, global_int value is: 8000000
  13. expected value is: 8000000
通过测试结果可以看出:
 
        1. 不加锁的情况下,不能获得正确结果。

测试结果表明,正确结果为8000000,而实际为1743884。表明多线程下修改全局计数,不加锁的话是错误的;

2. 加锁情况下,无论是线程锁还是原子性操作,均可获得正确结果。

3. 性能上__sync_fetch_and_add()完爆线程锁。

从性能测试结果上看,__sync_fetch_and_add()速度大致是线程锁的4-5倍。

测试结果对比
类型 平均CPU周期(circle) 平均耗时(us)
不加锁 180890066 75449.13
线程锁 10054091901 4193740.875
原子操作 2483427906 1035881.25

注:如上的性能测试结果,表明__sync_fetch_and_add()速度大致是线程锁的4-5倍,而并非文献【1】中6-7倍。由此,怀疑可能是由不同机器、不同CPU导致的,上述测试是在一台8core的虚拟机上实验的。为此,我又在不同的机器上重复相同的测试。

24cores实体机测试结果,表明__sync_fetch_and_add()速度大致只有线程锁的2-3倍。

24 cores实体机测试结果
类型 平均CPU周期(circle) 平均耗时(us)
不加锁 535457026 233310.5
线程锁 9331915480 4066156.667
原子操作 3769900795 1643463.625

总体看来,原子操作__sync_fetch_and_add()大大的优于线程锁。

另外:

上面介绍的原子操作参数里都有可扩展参数(...)用来指出哪些变量需要memory barrier,因为目前gcc实现的是full barrier(类似Linux kernel中的mb(),表示这个操作之前的所有内存操作不会被重排到这个操作之后),所以可以忽略掉这个参数。下面是有关memory barrier的东西。

关于memory barrier, cpu会对我们的指令进行排序,一般说来会提高程序的效率,但有时候可能造成我们不希望看到的结果。举例说明,比如我们有一硬件设备,当你发出一个操作指令的时候,一个寄存器存的是你的操作指令(READ),两个寄存器存的是参数(比如地址和size),最后一个寄存器是控制寄存器,在所有的参数都设置好后向其发出指令,设备开始读取参数,执行命令,程序可能如下:
             write1(dev.register_size, size);
             write1(dev.register_addr, addr);
             write1(dev.register_cmd, READ);
             write1(dev.register_control, Go);

       如果CPU对我们的指令进行优化排序,导致最后一条write1被换到前几条语句之前,那么肯定不是我们所期望的,这时候我们可以在最后一条语句之前加入一个memory barrier,强制CPU执行完前面的写入后再执行最后一条:
             write1(dev.register_size, size);
             write1(dev.register_addr, addr);
             write1(dev.register_cmd, READ);
             __sync_synchronize();            发出一个full barrier
             write1(dev.register_control, GO);
 
memory barrier有几种类型:
      acquire barrier:不允许将barrier之后的内存读取指令移到barrier之前;(linux kernel中的wmb)
      release barrier:不允许将barrier之前的内存读取指令移到barrier之后;(linux kernel中的rmb)
      full barrier:以上两种barrier的合集;(linux kernel中的mb)
 
参考文献:

[充电]多线程无锁编程--原子计数操作:__sync_fetch_and_add等12个操作的更多相关文章

  1. 【多线程】无锁编程以及CAS

    无锁编程 / lock-free / 非阻塞同步 无锁编程,即不使用锁的情况下实现多线程之间的变量同步,也就是在没有线程被阻塞的情况下实现变量的同步,所以也叫非阻塞同步(Non-blocking Sy ...

  2. [转]透过 Linux 内核看无锁编程

    非阻塞型同步 (Non-blocking Synchronization) 简介 如何正确有效的保护共享数据是编写并行程序必须面临的一个难题,通常的手段就是同步.同步可分为阻塞型同步(Blocking ...

  3. C++11原子操作与无锁编程(转)

    不讲语言特性,只从工程角度出发,个人觉得C++标准委员会在C++11中对多线程库的引入是有史以来做得最人道的一件事:今天我将就C++11多线程中的atomic原子操作展开讨论:比较互斥锁,自旋锁(sp ...

  4. C++性能榨汁机之无锁编程

    C++性能榨汁机之无锁编程 来源 http://irootlee.com/juicer_lock_free/ 前言 私以为个人的技术水平应该是一个螺旋式上升的过程:先从书本去了解一个大概,然后在实践中 ...

  5. 我是如何一步步的在并行编程中将lock锁次数降到最低实现无锁编程

    在并行编程中,经常会遇到多线程间操作共享集合的问题,很多时候大家都很难逃避这个问题做到一种无锁编程状态,你也知道一旦给共享集合套上lock之后,并发和伸缩能力往往会造成很大影响,这篇就来谈谈如何尽可能 ...

  6. 无锁编程以及CAS

    无锁编程 / lock-free / 非阻塞同步 无锁编程,即不使用锁的情况下实现多线程之间的变量同步,也就是在没有线程被阻塞的情况下实现变量的同步,所以也叫非阻塞同步(Non-blocking Sy ...

  7. 4.锁--无锁编程以及CAS

    无锁编程以及CAS 无锁编程 / lock-free / 非堵塞同步 无锁编程,即不使用锁的情况下实现多线程之间的变量同步,也就是在没有线程被堵塞的情况下实现变量的同步,所以也叫非堵塞同步(Non-b ...

  8. 无锁编程 - Double-checked Locking

    Double-checked Locking,严格意义上来讲不属于无锁范畴,无论什么时候当临界区中的代码仅仅需要加锁一次,同时当其获取锁的时候必须是线程安全的,此时就可以利用 Double-check ...

  9. 无锁编程(一) - Double-checked Locking

      Double-checked Locking,严格意义上来讲不属于无锁范畴,无论什么时候当临界区中的代码仅仅需要加锁一次,同时当其获取锁的时候必须是线程安全的,此时就可以利用 Double-che ...

随机推荐

  1. sbt commands

    速查手册 常用命令 actions – 显示对当前工程可用的命令 update – 下载依赖 compile – 编译代码 test – 运行测试代码 package – 创建一个可发布的jar包 p ...

  2. FK JavaScript之:ArcGIS JavaScript添加Graphic,地图界面却不显示

    使用ArcGIS JavaScript,往地图中添加几个Graphic,基本是与官网示例代码一致.绘制的图形一闪而过之后,就没了 核心代码如下: iniToolBar: function () { t ...

  3. 最新php一句话木马

    <?php $uf=”snc3″; $ka=”IEBldmFbsK”; $pjt=”CRfUE9TVF”; $vbl = str_replace(“ti”,””,”tistittirti_rti ...

  4. BFC以及文档流

    在一个文档流中,盒子模型元素的位置会互相影响. 当一个BFC出现在文档流中时,BFC内部的盒子模型元素同BFC外部的元素之间的位置不会互相影响. 相当于BFC重新创建了一个文档流. 举例: 一个文档流 ...

  5. LNMP环境简易安装流程

    1.关闭防火墙 [root@CentOS ~]# chkconfig iptables off 2.关闭selinux vi /etc/sysconfig/selinux //将SELINUX=enf ...

  6. MVC下载文件方式

    MVC下载文件方式 http://www.cnblogs.com/liang--liang/archive/2012/10/20/2732745.html 方式一: public FileStream ...

  7. TCP/IP协议 三次握手与四次挥手

    一.TCP报文格式 TCP/IP协议的详细信息参看<TCP/IP协议详解>三卷本.下面是TCP报文格式图: 图1 TCP报文格式 上图中有几个字段需要重点介绍下:        (1)序号 ...

  8. tcpdump教程入门

    tcpdump是一个最基本重要的网络分析工具, 掌握好这, 对于学习tcp/ip协议也是很有帮助的. 理解了tcp/ip协议栈的知识, 分析调优网络的能力才会更高. 所以使用tcpdump相比其它的工 ...

  9. 【转】Web性能压力测试工具之ApacheBench(ab)详解

    PS:网站性能压力测试是性能调优过程中必不可少的一环.只有让服务器处在高压情况下才能真正体现出各种设置所暴露的问题.Apache中有个自带的,名为ab的程序,可以对Apache或其它类型的服务器进行网 ...

  10. Fiddler---前端利器

    摘要:Fiddler是一个不错的工具,可以用来调试HTTP协议.工具很强大,本篇根据现有使用总结了下,以后有更多使用再做补充. 在做HTTPS协议对接的时候,拿不到数据,不得不找个工具试下,确定是接收 ...