亚洲av成人无遮挡网站在线观看,少妇性bbb搡bbb爽爽爽,亚洲av日韩精品久久久久久,兔费看少妇性l交大片免费,无码少妇一区二区三区

  免費(fèi)注冊(cè) 查看新帖 |

Chinaunix

  平臺(tái) 論壇 博客 文庫
最近訪問板塊 發(fā)新帖
查看: 42364 | 回復(fù): 3
打印 上一主題 下一主題

[C++] C++ atomic compare_and_exchange方法困惑 [復(fù)制鏈接]

論壇徽章:
2
2015年辭舊歲徽章
日期:2015-03-03 16:54:152015元宵節(jié)徽章
日期:2015-03-06 15:52:30
跳轉(zhuǎn)到指定樓層
1 [收藏(0)] [報(bào)告]
發(fā)表于 2016-07-27 12:38 |只看該作者 |倒序?yàn)g覽
我知道gcc提供了如下內(nèi)建函數(shù),
__sync_bool_compare_and_swap, 在x86上會(huì)編譯成 lock cmpxchg指令
可對(duì)內(nèi)存進(jìn)行原子訪問。

這個(gè)函數(shù)的原型是
bool __sync_bool_compare_and_swap (type *ptr, type oldval type newval, ...)
type是整數(shù),比如int。

如果用C來表示這個(gè)函數(shù)做的事情,應(yīng)該是這樣的:


  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. }

復(fù)制代碼
這也是通常的CAS實(shí)現(xiàn)的語義。這個(gè)函數(shù)可以用來實(shí)現(xiàn)無鎖。

在C++11中,有一個(gè)atomic庫,里面的atomic<T>模板類,
其中有方法叫 compare_exchange_weak/strong.
關(guān)于weak和strong的差別我理解了,可以看這里 http://www.cplusplus.com/referen ... are_exchange_strong。

我的問題是,這個(gè)方法的語義看起來和 CAS不太一樣。
這個(gè)方法的運(yùn)行結(jié)果用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. }

復(fù)制代碼
我看了后很困惑,我查了一些資料,也是說這個(gè)方法是用來實(shí)現(xiàn)無鎖的,通常放在一個(gè)while中循環(huán)直到返回true.
問題是,這個(gè)方法如果第一次調(diào)用是false的話,第二次調(diào)用肯定是true了。

論壇徽章:
2
2015年辭舊歲徽章
日期:2015-03-03 16:54:152015元宵節(jié)徽章
日期:2015-03-06 15:52:30
2 [報(bào)告]
發(fā)表于 2016-07-27 15:32 |只看該作者
回復(fù) 1# neodreamerus


網(wǎng)上收了一陣后有點(diǎn)明白了, std::atomic 實(shí)現(xiàn)的CAS 和 __sync_bool_compare_and_swap確實(shí)有點(diǎn)不一樣,
但是能實(shí)現(xiàn)相同的效果。

找到一個(gè)使用 std::atomic 實(shí)現(xiàn)無鎖鏈表的簡(jiǎn)單例子。學(xué)習(xí)一下。


  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. }

復(fù)制代碼

論壇徽章:
0
3 [報(bào)告]
發(fā)表于 2016-07-28 00:55 |只看該作者
就是一個(gè)返回最新值,一個(gè)沒有返回的區(qū)別,返回的話,后續(xù)可以少一個(gè)load操作,也許但不一定有加快速度的幾率。

論壇徽章:
14
巨蟹座
日期:2013-11-19 14:09:4615-16賽季CBA聯(lián)賽之青島
日期:2016-07-05 12:36:0515-16賽季CBA聯(lián)賽之廣東
日期:2016-06-29 11:45:542015亞冠之全北現(xiàn)代
日期: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技術(shù)圖書徽章
日期:2014-04-15 08:44:01天蝎座
日期:2014-03-11 13:06:45
4 [報(bào)告]
發(fā)表于 2016-07-28 09:18 |只看該作者
第二個(gè)實(shí)現(xiàn)最好吧
第一個(gè)實(shí)現(xiàn),在失敗后,你還得手工重置oldval=*ptr,否則以后全是失敗
您需要登錄后才可以回帖 登錄 | 注冊(cè)

本版積分規(guī)則 發(fā)表回復(fù)

  

北京盛拓優(yōu)訊信息技術(shù)有限公司. 版權(quán)所有 京ICP備16024965號(hào)-6 北京市公安局海淀分局網(wǎng)監(jiān)中心備案編號(hào):11010802020122 niuxiaotong@pcpop.com 17352615567
未成年舉報(bào)專區(qū)
中國互聯(lián)網(wǎng)協(xié)會(huì)會(huì)員  聯(lián)系我們:huangweiwei@itpub.net
感謝所有關(guān)心和支持過ChinaUnix的朋友們 轉(zhuǎn)載本站內(nèi)容請(qǐng)注明原作者名及出處

清除 Cookies - ChinaUnix - Archiver - WAP - TOP