欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

探究在C++程序并發(fā)時(shí)保護(hù)共享數(shù)據(jù)的問題

 更新時(shí)間:2015年07月14日 11:49:55   投稿:goldensun  
這篇文章主要介紹了探究在C++程序并發(fā)時(shí)保護(hù)共享數(shù)據(jù)的問題,也有利于大家更好地理解C++多線程的一些機(jī)制,需要的朋友可以參考下

 我們先通過一個(gè)簡(jiǎn)單的代碼來了解該問題。
同步問題

我們使用一個(gè)簡(jiǎn)單的結(jié)構(gòu)體 Counter,該結(jié)構(gòu)體包含一個(gè)值以及一個(gè)方法用來改變這個(gè)值:
 

struct Counter {
  int value;
 
  void increment(){
    ++value;
  }
};

然后啟動(dòng)多個(gè)線程來修改結(jié)構(gòu)體的值:

 

int main(){
  Counter counter;
 
  std::vector<std::thread> threads;
  for(int i = 0; i < 5; ++i){
    threads.push_back(std::thread([&counter](){
      for(int i = 0; i < 100; ++i){
        counter.increment();
      }
    }));
  }
 
  for(auto& thread : threads){
    thread.join();
  }
 
  std::cout << counter.value << std::endl;
 
  return 0;
}

我們啟動(dòng)了5個(gè)線程來增加計(jì)數(shù)器的值,每個(gè)線程增加了100次,然后在線程結(jié)束時(shí)打印計(jì)數(shù)器的值。


但我們運(yùn)行這個(gè)程序的時(shí)候,我們是希望它會(huì)答應(yīng)500,但事實(shí)不是如此,沒人能確切知道程序?qū)⒋蛴∈裁唇Y(jié)果,下面是在我機(jī)器上運(yùn)行后打印的數(shù)據(jù),而且每次都不同:
 

442
500
477
400
422
487

問題的原因在于改變計(jì)數(shù)器值并不是一個(gè)原子操作,需要經(jīng)過下面三個(gè)操作才能完成一次計(jì)數(shù)器的增加:

  •     首先讀取 value 的值
  •     然后將 value 值加1
  •     將新的值賦值給 value

但你使用單線程來運(yùn)行這個(gè)程序的時(shí)候當(dāng)然沒有任何問題,因此程序是順序執(zhí)行的,但在多線程環(huán)境中就有麻煩了,想象下下面這個(gè)執(zhí)行順序:

  •     Thread 1 : 讀取 value, 得到 0, 加 1, 因此 value = 1
  •     Thread 2 : 讀取 value, 得到 0, 加 1, 因此 value = 1
  •     Thread 1 : 將 1 賦值給 value,然后返回 1
  •     Thread 2 : 將 1 賦值給 value,然后返回 1

這種情況我們稱之為多線程的交錯(cuò)執(zhí)行,也就是說多線程可能在同一個(gè)時(shí)間點(diǎn)執(zhí)行相同的語句,盡管只有兩個(gè)線程,交錯(cuò)的現(xiàn)象也很明顯。如果你有更多的線程、更多的操作需要執(zhí)行,那么這個(gè)交錯(cuò)是必然發(fā)生的。

有很多方法來解決線程交錯(cuò)的問題:

  •     信號(hào)量 Semaphores
  •     原子引用 Atomic references
  •     Monitors
  •     Condition codes
  •     Compare and swap

在這篇文章中我們將學(xué)習(xí)如何使用信號(hào)量來解決這個(gè)問題。信號(hào)量也有很多人稱之為互斥量(Mutex),同一個(gè)時(shí)間只允許一個(gè)線程獲取一個(gè)互斥對(duì)象的鎖,通過 Mutex 的簡(jiǎn)單屬性就可以用來解決交錯(cuò)的問題。

使用 Mutex 讓計(jì)數(shù)器程序是線程安全的

在 C++11 線程庫中,互斥量包含在 mutex 頭文件中,對(duì)應(yīng)的類是 std::mutex,有兩個(gè)重要的方法 mutex:lock() 和 unlock() ,從名字上可得知是用來鎖對(duì)象以及釋放鎖對(duì)象。一旦某個(gè)互斥量被鎖,那么再次調(diào)用 lock() 返回堵塞值得該對(duì)象被釋放。

為了讓我們剛才的計(jì)數(shù)器結(jié)構(gòu)體是線程安全的,我們添加一個(gè) set:mutext 成員,并在每個(gè)方法中通過 lock()/unlock() 方法來進(jìn)行保護(hù):
 

struct Counter {
  std::mutex mutex;
  int value;
 
  Counter() : value(0) {}
 
  void increment(){
    mutex.lock();
    ++value;
    mutex.unlock();
  }
};

然后我們?cè)俅螠y(cè)試這個(gè)程序,打印的結(jié)果就是 500 了,而且每次都一樣。

異常和鎖

現(xiàn)在讓我們來看另外一種情況,想象我們的的計(jì)數(shù)器有一個(gè)減操作,并在值為0的時(shí)候拋出異常:
 

struct Counter {
  int value;
 
  Counter() : value(0) {}
 
  void increment(){
    ++value;
  }
 
  void decrement(){
    if(value == 0){
      throw "Value cannot be less than 0";
    }
 
    --value;
  }
};

然后我們不需要修改類來訪問這個(gè)結(jié)構(gòu)體,我們創(chuàng)建一個(gè)封裝器:
 

struct ConcurrentCounter {
  std::mutex mutex;
  Counter counter;
 
  void increment(){
    mutex.lock();
    counter.increment();
    mutex.unlock();
  }
 
  void decrement(){
    mutex.lock();
    counter.decrement();    
    mutex.unlock();
  }
};

大部分時(shí)候該封裝器運(yùn)行挺好,但是使用 decrement 方法的時(shí)候就會(huì)有異常發(fā)生。這是一個(gè)大問題,一旦異常發(fā)生后,unlock 方法就沒被調(diào)用,導(dǎo)致互斥量一直被占用,然后整個(gè)程序就一直處于堵塞狀態(tài)(死鎖),為了解決這個(gè)問題我們需要用 try/catch 結(jié)構(gòu)來處理異常情況:
 

void decrement(){
  mutex.lock();
  try {
    counter.decrement();
  } catch (std::string e){
    mutex.unlock();
    throw e;
  }
  mutex.unlock();
}

這個(gè)代碼并不難,但看起來很丑,如果你一個(gè)函數(shù)有 10 個(gè)退出點(diǎn),你就必須為每個(gè)退出點(diǎn)調(diào)用一次 unlock 方法,或許你可能在某個(gè)地方忘掉了 unlock ,那么各種悲劇即將發(fā)生,悲劇發(fā)生將直接導(dǎo)致程序死鎖。

接下來我們看如何解決這個(gè)問題。

自動(dòng)鎖管理

當(dāng)你需要包含整段的代碼(在我們這里是一個(gè)方法,也可能是一個(gè)循環(huán)體或者其他的控制結(jié)構(gòu)),有這么一種好的解決方法可以避免忘記釋放鎖,那就是 std::lock_guard.

這個(gè)類是一個(gè)簡(jiǎn)單的智能鎖管理器,但創(chuàng)建 std::lock_guard 時(shí),會(huì)自動(dòng)調(diào)用互斥量對(duì)象的 lock() 方法,當(dāng) lock_guard 析構(gòu)時(shí)會(huì)自動(dòng)釋放鎖,請(qǐng)看下面代碼:

 

struct ConcurrentSafeCounter {
  std::mutex mutex;
  Counter counter;
 
  void increment(){
    std::lock_guard<std::mutex> guard(mutex);
    counter.increment();
  }
 
  void decrement(){
    std::lock_guard<std::mutex> guar(mutex);
    mutex.unlock();
  }
};

是不是看起來爽多了?

使用 lock_guard ,你不再需要考慮什么時(shí)候要釋放鎖,這個(gè)工作已經(jīng)由 std::lock_guard 實(shí)例幫你完成。

結(jié)論

在這篇文章中我們學(xué)習(xí)了如何通過信號(hào)量/互斥量來保護(hù)共享數(shù)據(jù)。需要記住的是,使用鎖會(huì)降低程序性能。在一些高并發(fā)的應(yīng)用環(huán)境中有其他更好的解決辦法,不過這不在本文的討論范疇之內(nèi)。

你可以在 Github 上獲取本文的源碼.

相關(guān)文章

  • C++內(nèi)存模型和名稱空間詳解

    C++內(nèi)存模型和名稱空間詳解

    這篇文章主要給大家介紹了關(guān)于C/C++中的內(nèi)存模型和名稱空間詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用c/c++具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起看看吧
    2021-09-09
  • C語言逆向分析語法超詳細(xì)分析

    C語言逆向分析語法超詳細(xì)分析

    這篇文章主要介紹了C語言逆向分析語法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2022-11-11
  • C語言實(shí)現(xiàn)多線程定時(shí)器實(shí)例講解

    C語言實(shí)現(xiàn)多線程定時(shí)器實(shí)例講解

    在本篇文章里小編給各位分享的是一篇關(guān)于C語言實(shí)現(xiàn)多線程定時(shí)器實(shí)例講解內(nèi)容,有需要的朋友們可以參考學(xué)習(xí)下。
    2021-01-01
  • C++編程歸并排序算法實(shí)現(xiàn)示例

    C++編程歸并排序算法實(shí)現(xiàn)示例

    這篇文章主要為大家介紹了C++編程歸并排序算法實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪
    2021-11-11
  • c++11 chrono全面解析(最高可達(dá)納秒級(jí)別的精度)

    c++11 chrono全面解析(最高可達(dá)納秒級(jí)別的精度)

    chrono是c++ 11中的時(shí)間庫,本文就來詳細(xì)的介紹一下chrono庫的具體使用,關(guān)鍵是理解里面時(shí)間段(Durations)、時(shí)間點(diǎn)(Time points)的概念,感興趣的可以了解一下
    2021-11-11
  • 深入分析C++中deque的使用

    深入分析C++中deque的使用

    本篇文章介紹了,深入分析C++中deque的使用。需要的朋友參考下
    2013-05-05
  • C++設(shè)計(jì)模式之模板方法模式(TemplateMethod)

    C++設(shè)計(jì)模式之模板方法模式(TemplateMethod)

    這篇文章主要為大家詳細(xì)介紹了C++設(shè)計(jì)模式之模板方法模式TemplateMethod,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-04-04
  • C++實(shí)現(xiàn)MyString的示例代碼

    C++實(shí)現(xiàn)MyString的示例代碼

    本文主要介紹了C++實(shí)現(xiàn)MyString的示例代碼,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • 詳解C語言解決經(jīng)典問題之兔子產(chǎn)子

    詳解C語言解決經(jīng)典問題之兔子產(chǎn)子

    有一對(duì)兔子,從出生后的第 3 個(gè)月起每個(gè)月都生一對(duì)兔子。小兔子長到第 3 個(gè)月后每個(gè)月又生一對(duì)兔子,假設(shè)所有的兔子都不死,問 30 個(gè)月內(nèi)每個(gè)月的兔子總數(shù)為多少?本文將用C語言解決這一經(jīng)典問題,需要的可以參考一下
    2022-03-03
  • C++分析如何用虛析構(gòu)與純虛析構(gòu)處理內(nèi)存泄漏

    C++分析如何用虛析構(gòu)與純虛析構(gòu)處理內(nèi)存泄漏

    虛析構(gòu)和純虛析構(gòu)共性:可以解決父類指針釋放子類對(duì)象,都需要有具體的函數(shù)實(shí)現(xiàn);虛析構(gòu)和純虛析構(gòu)區(qū)別:如果是純虛析構(gòu),該類屬于抽象類,無法實(shí)例化對(duì)象
    2022-08-08

最新評(píng)論