免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 42195 | 回复: 3

[C++] C++ atomic compare_and_exchange方法困惑 [复制链接]

论坛徽章:
2
2015年辞旧岁徽章
日期:2015-03-03 16:54:152015元宵节徽章
日期:2015-03-06 15:52:30
发表于 2016-07-27 12:38 |显示全部楼层
我知道gcc提供了如下内建函数,
__sync_bool_compare_and_swap, 在x86上会编译成 lock cmpxchg指令
可对内存进行原子访问。

这个函数的原型是
bool __sync_bool_compare_and_swap (type *ptr, type oldval type newval, ...)
type是整数,比如int。

如果用C来表示这个函数做的事情,应该是这样的:


  1. bool  __sync_bool_compare_and_swap(int* ptr, int oldval, int newval) {

  2.      if (*ptr == oldval) {
  3.            *ptr = newval;
  4.             return true;
  5.      } else return false;
  6. }

复制代码
这也是通常的CAS实现的语义。这个函数可以用来实现无锁。

在C++11中,有一个atomic库,里面的atomic<T>模板类,
其中有方法叫 compare_exchange_weak/strong.
关于weak和strong的差别我理解了,可以看这里 http://www.cplusplus.com/referen ... are_exchange_strong

我的问题是,这个方法的语义看起来和 CAS不太一样。
这个方法的运行结果用C表示大概是这样的:


  1. bool  atomic_compare_exchange(int* ptr, int* oldval, int newval) {

  2.      if (*ptr == *oldval) {
  3.            *ptr = newval;
  4.             return true;
  5.      } else {
  6.             *oldval = *ptr;
  7.             return false;
  8.      }
  9. }

复制代码
我看了后很困惑,我查了一些资料,也是说这个方法是用来实现无锁的,通常放在一个while中循环直到返回true.
问题是,这个方法如果第一次调用是false的话,第二次调用肯定是true了。

论坛徽章:
2
2015年辞旧岁徽章
日期:2015-03-03 16:54:152015元宵节徽章
日期:2015-03-06 15:52:30
发表于 2016-07-27 15:32 |显示全部楼层
回复 1# neodreamerus


网上收了一阵后有点明白了, std::atomic 实现的CAS 和 __sync_bool_compare_and_swap确实有点不一样,
但是能实现相同的效果。

找到一个使用 std::atomic 实现无锁链表的简单例子。学习一下。


  1. #include <atomic>

  2. template<typename T>
  3. class List {
  4. public:
  5.   void append(const T& value);   // see below

  6. private:
  7.   struct Node {
  8.     T data;
  9.     Node *next;
  10.   };  

  11.   std::atomic<Node*> head;
  12. };

  13. template<typename T>
  14. void List<T>::append(const T& value)
  15. {
  16.   Node *newNode = new Node { value };
  17.    
  18.   Node* headNode;
  19.   do {
  20.     headNode = head;
  21.     newNode->next = head;
  22.   } while (!std::atomic_compare_exchange_weak(&head, &headNode, newNode));
  23.   // or while (!head.compare_exchange_weak(headNode, newNode));
  24. }

  25. int main()
  26. {
  27.   List<int> li;
  28.   li.append(10);
  29. }

复制代码

论坛徽章:
0
发表于 2016-07-28 00:55 |显示全部楼层
就是一个返回最新值,一个没有返回的区别,返回的话,后续可以少一个load操作,也许但不一定有加快速度的几率。

论坛徽章:
14
巨蟹座
日期:2013-11-19 14:09:4615-16赛季CBA联赛之青岛
日期:2016-07-05 12:36:0515-16赛季CBA联赛之广东
日期:2016-06-29 11:45:542015亚冠之全北现代
日期:2015-07-22 08:09:472015年辞旧岁徽章
日期:2015-03-03 16:54:15巨蟹座
日期:2014-12-29 08:22:29射手座
日期:2014-12-05 08:20:39狮子座
日期:2014-11-05 12:33:52寅虎
日期:2014-08-13 09:01:31巳蛇
日期:2014-06-16 16:29:52技术图书徽章
日期:2014-04-15 08:44:01天蝎座
日期:2014-03-11 13:06:45
发表于 2016-07-28 09:18 |显示全部楼层
第二个实现最好吧
第一个实现,在失败后,你还得手工重置oldval=*ptr,否则以后全是失败
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP