真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

C++11多線程編程的方法

本篇內(nèi)容介紹了“C++11多線程編程的方法”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

為西青等地區(qū)用戶提供了全套網(wǎng)頁設(shè)計制作服務(wù),及西青網(wǎng)站建設(shè)行業(yè)解決方案。主營業(yè)務(wù)為網(wǎng)站建設(shè)、成都網(wǎng)站設(shè)計、西青網(wǎng)站設(shè)計,以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務(wù)。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!

1.在C++11中創(chuàng)建新線程

在每個c++應(yīng)用程序中,都有一個默認的主線程,即main函數(shù),在c++11中,我們可以通過創(chuàng)建std::thread類的對象來創(chuàng)建其他線程,每個std :: thread對象都可以與一個線程相關(guān)聯(lián),只需包含頭文件< thread>??梢允褂胹td :: thread對象附加一個回調(diào),當這個新線程啟動時,它將被執(zhí)行。 這些回調(diào)可以為函數(shù)指針、函數(shù)對象、Lambda函數(shù)。
線程對象可通過std::thread thObj(< CALLBACK>)來創(chuàng)建,新線程將在創(chuàng)建新對象后立即開始,并且將與已啟動的線程并行執(zhí)行傳遞的回調(diào)。此外,任何線程可以通過在該線程的對象上調(diào)用join()函數(shù)來等待另一個線程退出。
使用函數(shù)指針創(chuàng)建線程:

//main.cpp
#include 
#include 
void thread_function() {    
    for (int i = 0; i < 5; i++)        
    std::cout << "thread function excuting" << std::endl;
}int main() {    
    std::thread threadObj(thread_function);    
    for (int i = 0; i < 5; i++)        
    std::cout << "Display from MainThread" << std::endl;
   threadObj.join();    
   std::cout << "Exit of Main function" << std::endl;    return 0;
}

使用函數(shù)對象創(chuàng)建線程:

#include 
#include 
class DisplayThread {
    public:void operator ()() {        
        for (int i = 0; i < 100; i++)            
        std::cout << "Display Thread Excecuting" << std::endl;
    }
};
int main() {    
    std::thread threadObj((DisplayThread()));    
    for (int i = 0; i < 100; i++)        
    std::cout << "Display From Main Thread " << std::endl;    
    std::cout << "Waiting For Thread to complete" << std::endl;
    threadObj.join();    
    std::cout << "Exiting from Main Thread" << std::endl;    
    return 0;
}

CmakeLists.txt

cmake_minimum_required(VERSION 3.10)
project(Thread_test)set(CMAKE_CXX_STANDARD 11)
find_package(Threads REQUIRED)
add_executable(Thread_test main.cpp)
target_link_libraries(Thread_test ${CMAKE_THREAD_LIBS_INIT})

每個std::thread對象都有一個相關(guān)聯(lián)的id,std::thread::get_id() —-成員函數(shù)中給出對應(yīng)線程對象的id;
std::this_thread::get_id()—-給出當前線程的id,如果std::thread對象沒有關(guān)聯(lián)的線程,get_id()將返回默認構(gòu)造的std::thread::id對象:“not any thread”,std::thread::id也可以表示id。

2.joining和detaching 線程

線程一旦啟動,另一個線程可以通過調(diào)用std::thread對象上調(diào)用join()函數(shù)等待這個線程執(zhí)行完畢:

std::thread threadObj(funcPtr); 
threadObj.join();

例如,主線程啟動10個線程,啟動完畢后,main函數(shù)等待他們執(zhí)行完畢,join完所有線程后,main函數(shù)繼續(xù)執(zhí)行:

#include 
#include 
#include 
class WorkerThread
{
    public:void operator()(){        
        std::cout<<"Worker Thread "< threadList;    
        for(int i = 0; i < 10; i++){
        threadList.push_back(std::thread(WorkerThread()));
    }    
    // Now wait for all the worker thread to finish i.e.
    // Call join() function on each of the std::thread object
    std::cout<<"Wait for all the worker thread to finish"<

detach可以將線程與線程對象分離,讓線程作為后臺線程執(zhí)行,當前線程也不會阻塞了.但是detach之后就無法在和線程發(fā)生聯(lián)系了.如果線程執(zhí)行函數(shù)使用了臨時變量可能會出現(xiàn)問題,線程調(diào)用了detach在后臺運行,臨時變量可能已經(jīng)銷毀,那么線程會訪問已經(jīng)被銷毀的變量,需要在std::thread對象中調(diào)用std::detach()函數(shù):

std::thread threadObj(funcPtr)
threadObj.detach();

調(diào)用detach()后,std::thread對象不再與實際執(zhí)行線程相關(guān)聯(lián),在線程句柄上調(diào)用detach() 和 join()要小心.

3.將參數(shù)傳遞給線程

要將參數(shù)傳遞給線程的可關(guān)聯(lián)對象或函數(shù),只需將參數(shù)傳遞給std::thread構(gòu)造函數(shù),默認情況下,所有的參數(shù)都將復(fù)制到新線程的內(nèi)部存儲中。
給線程傳遞參數(shù):

#include 
#include 
#include 
void threadCallback(int x, std::string str) {  
    std::cout << "Passed Number = " << x << std::endl;  
    std::cout << "Passed String = " << str << std::endl;
}int main() {  
    int x = 10;  
    std::string str = "Sample String";  
    std::thread threadObj(threadCallback, x, str);
  threadObj.join();  
  return 0;
}

給線程傳遞引用:

#include 
#include 
void threadCallback(int const& x) {  
    int& y = const_cast(x);
  y++;  
  std::cout << "Inside Thread x = " << x << std::endl;
}int main() {  
    int x = 9;  
    std::cout << "In Main Thread : Before Thread Start x = " << x << std::endl;  
    std::thread threadObj(threadCallback, x);
  threadObj.join();  
  std::cout << "In Main Thread : After Thread Joins x = " << x << std::endl;  
  return 0;
}

輸出結(jié)果為:
In Main Thread : Before Thread Start x = 9
Inside Thread x = 10
In Main Thread : After Thread Joins x = 9

Process finished with exit code 0
即使threadCallback接受參數(shù)作為引用,但是并沒有改變main中x的值,在線程引用外它是不可見的。這是因為線程函數(shù)threadCallback中的x是引用復(fù)制在新線程的堆棧中的臨時值,使用std::ref可進行修改:

#include 
#include 
void threadCallback(int const& x) {  
    int& y = const_cast(x);
  y++;  
  std::cout << "Inside Thread x = " << x << std::endl;
}int main() {  
    int x = 9;  std::cout << "In Main Thread : Before Thread Start x = " << x << std::endl;  
    std::thread threadObj(threadCallback, std::ref(x));
  threadObj.join();  
  std::cout << "In Main Thread : After Thread Joins x = " << x << std::endl;  
  return 0;
}

輸出結(jié)果為:
In Main Thread : Before Thread Start x = 9
Inside Thread x = 10
In Main Thread : After Thread Joins x = 10

Process finished with exit code 0
指定一個類的成員函數(shù)的指針作為線程函數(shù),將指針傳遞給成員函數(shù)作為回調(diào)函數(shù),并將指針指向?qū)ο笞鳛榈诙€參數(shù):

#include 
#include 
class DummyClass { 
public:
  DummyClass() { }
  DummyClass(const DummyClass& obj) { }  
  void sampleMemberfunction(int x) {    
      std::cout << "Inside sampleMemberfunction " << x << std::endl;
  }
};
    int main() {
      DummyClass dummyObj;  
      int x = 10;  
      std::thread threadObj(&DummyClass::sampleMemberfunction, &dummyObj, x);
      threadObj.join();  
      return 0;
}

4.線程間數(shù)據(jù)的共享與競爭條件

在多線程間的數(shù)據(jù)共享很簡單,但是在程序中的這種數(shù)據(jù)共享可能會引起問題,其中一種便是競爭條件。當兩個或多個線程并行執(zhí)行一組操作,訪問相同的內(nèi)存位置,此時,它們中的一個或多個線程會修改內(nèi)存位置中的數(shù)據(jù),這可能會導(dǎo)致一些意外的結(jié)果,這就是競爭條件。競爭條件通常較難發(fā)現(xiàn)并重現(xiàn),因為它們并不總是出現(xiàn),只有當兩個或多個線程執(zhí)行操作的相對順序?qū)е乱馔饨Y(jié)果時,它們才會發(fā)生。
例如創(chuàng)建5個線程,這些線程共享類Wallet的一個對象,使用addMoney()成員函數(shù)并行添加100元。所以,如果最初錢包中的錢是0,那么在所有線程的競爭執(zhí)行完畢后,錢包中的錢應(yīng)該是500,但是,由于所有線程同時修改共享數(shù)據(jù),在某些情況下,錢包中的錢可能遠小于500。
測試如下:

#include 
#include 
#include 
class Wallet {    
    int mMoney;
    public: Wallet() : mMoney(0) { }    
    int getMoney() { return mMoney; }    
    void addMoney(int money) {        
        for (int i = 0; i < money; i++) {
            mMoney++;
        }
    }
};int testMultithreadWallet() {
    Wallet walletObject;    
    std::vector threads;    
    for (int i = 0; i < 5; i++) {
        threads.push_back(std::thread(&Wallet::addMoney, &walletObject, 100));
    }    for (int i = 0; i < 5; i++) {
        threads.at(i).join();
    }    
    return walletObject.getMoney();
}int main() {    
        int val = 0;    
        for (int k = 0; k < 100; k++) {        
        if ((val=testMultithreadWallet()) != 500) {            
            std::cout << "Error at count = " << k << " Money in Wallet = " << val << std::endl;
        }
    }    
    return 0;
}

每個線程并行地增加相同的成員變量“mMoney”,看似是一條線,但是這個“nMoney++”實際上被轉(zhuǎn)換為3條機器命令:
·在Register中加載”mMoney”變量
·增加register的值
·用register的值更新“mMoney”變量
在這種情況下,一個增量將被忽略,因為不是增加mMoney變量,而是增加不同的寄存器,“mMoney”變量的值被覆蓋。

5.使用mutex處理競爭條件

為了處理多線程環(huán)境中的競爭條件,我們需要mutex互斥鎖,在修改或讀取共享數(shù)據(jù)前,需要對數(shù)據(jù)加鎖,修改完成后,對數(shù)據(jù)進行解鎖。在c++11的線程庫中,mutexes在< mutexe >頭文件中,表示互斥體的類是std::mutex。
就上面的問題進行處理,Wallet類提供了在Wallet中增加money的方法,并且在不同的線程中使用相同的Wallet對象,所以我們需要對Wallet的addMoney()方法加鎖。在增加Wallet中的money前加鎖,并且在離開該函數(shù)前解鎖,看代碼:Wallet類內(nèi)部維護money,并提供函數(shù)addMoney(),這個成員函數(shù)首先獲取一個鎖,然后給wallet對象的money增加指定的數(shù)額,最后釋放鎖。

#include 
#include 
#include 
#include 
class Wallet {    
        int mMoney;    
        std::mutex mutex;public:
    Wallet() : mMoney(0) { }    
    int getMoney() { return mMoney;}    
    void addMoney(int money) {
        mutex.lock();        
        for (int i = 0; i < money; i++) {
            mMoney++;
        }
        mutex.unlock();
    }
};int testMultithreadWallet() {
    Wallet walletObject;    
    std::vector threads;    
    for (int i = 0; i < 5; ++i) {
        threads.push_back(std::thread(&Wallet::addMoney, &walletObject, 1000));
    }    for (int i = 0; i < threads.size(); i++) {
        threads.at(i).join();
    }    
    return walletObject.getMoney();
}int main() {    
        int val = 0;    
        for (int k = 0; k < 1000; k++) {        
        if ((val = testMultithreadWallet()) != 5000) {            
        std::cout << "Error at count= " << k << " money in wallet" << val << std::endl;
        }
    }    
    return 0;
}

這種情況保證了錢包里的錢不會出現(xiàn)少于5000的情況,因為addMoney()中的互斥鎖確保了只有在一個線程修改完成money后,另一個線程才能對其進行修改,但是,如果我們忘記在函數(shù)結(jié)束后對鎖進行釋放會怎么樣?這種情況下,一個線程將退出而不釋放鎖,其他線程將保持等待,為了避免這種情況,我們應(yīng)當使用std::lock_guard,這是一個template class,它為mutex實現(xiàn)RALL,它將mutex包裹在其對象內(nèi),并將附加的mutex鎖定在其構(gòu)造函數(shù)中,當其析構(gòu)函數(shù)被調(diào)用時,它將釋放互斥體。

class Wallet {  
    int mMoney;  
    std::mutex mutex; public:
  Wallet() : mMoney(0) { }  int getMoney() { return mMoney;}  
  void addMoney(int money) {    
  std::lock_guard lockGuard(mutex);    
  for (int i = 0; i < mMoney; ++i) {      
  //如果在此處發(fā)生異常,lockGuadr的析構(gòu)函數(shù)將會因為堆棧展開而被調(diào)用
      mMoney++;      
      //一旦函數(shù)退出,那么lockGuard對象的析構(gòu)函數(shù)將被調(diào)用,在析構(gòu)函數(shù)中mutex會被釋放
    }

  }
};

6.條件變量

條件變量是一種用于在2個線程之間進行信令的事件,一個線程可以等待它得到信號,其他的線程可以給它發(fā)信號。在c++11中,條件變量需要頭文件< condition_variable>,同時,條件變量還需要一個mutex鎖。
條件變量是如何運行的:
·線程1調(diào)用等待條件變量,內(nèi)部獲取mutex互斥鎖并檢查是否滿足條件;
·如果沒有,則釋放鎖,并等待條件變量得到發(fā)出的信號(線程被阻塞),條件變量的wait()函數(shù)以原子方式提供這兩個操作;
·另一個線程,如線程2,當滿足條件時,向條件變量發(fā)信號;
·一旦線程1正等待其恢復(fù)的條件變量發(fā)出信號,線程1便獲取互斥鎖,并檢查與條件變量相關(guān)關(guān)聯(lián)的條件是否滿足,或者是否是一個上級調(diào)用,如果多個線程正在等待,那么notify_one將只解鎖一個線程;
·如果是一個上級調(diào)用,那么它再次調(diào)用wait()函數(shù)。
條件變量的主要成員函數(shù):
Wait()
它使得當前線程阻塞,直到條件變量得到信號或發(fā)生虛假喚醒;
它原子性地釋放附加的mutex,阻塞當前線程,并將其添加到等待當前條件變量對象的線程列表中,當某線程在同樣的條件變量上調(diào)用notify_one() 或者 notify_all(),線程將被解除阻塞;
這種行為也可能是虛假的,因此,解除阻塞后,需要再次檢查條件;
一個回調(diào)函數(shù)會傳給該函數(shù),調(diào)用它來檢查其是否是虛假調(diào)用,還是確實滿足了真實條件;
當線程解除阻塞后,wait()函數(shù)獲取mutex鎖,并檢查條件是否滿足,如果條件不滿足,則再次原子性地釋放附加的mutex,阻塞當前線程,并將其添加到等待當前條件變量對象的線程列表中。
notify_one()
如果所有線程都在等待相同的條件變量對象,那么notify_one會取消阻塞其中一個等待線程。
notify_all()
如果所有線程都在等待相同的條件變量對象,那么notify_all會取消阻塞所有的等待線程。

#include 
#include 
#include 
#include 
#include 
using namespace std::placeholders;
class Application {    
    std::mutex m_mutex;    
    std::condition_variable m_condVar;    
    bool m_bDataLoaded;public:
  Application() {
        m_bDataLoaded = false;
    }    
    void loadData() {        
            //使該線程sleep 1秒
        std::this_thread::sleep_for(std::chrono::milliseconds(1000));        
        std::cout << "Loading Data from XML" << std::endl;        //鎖定數(shù)據(jù)
        std::lock_guard guard(m_mutex);        //flag設(shè)為true,表明數(shù)據(jù)已加載
        m_bDataLoaded = true;        //通知條件變量
        m_condVar.notify_one();
    }    bool isDataLoaded() {        
             return m_bDataLoaded;
    }    void mainTask() {        
            std::cout << "Do some handshaking" << std::endl;        //獲取鎖
        std::unique_lock mlock(m_mutex);        //開始等待條件變量得到信號
        //wait()將在內(nèi)部釋放鎖,并使線程阻塞
        //一旦條件變量發(fā)出信號,則恢復(fù)線程并再次獲取鎖
        //然后檢測條件是否滿足,如果條件滿足,則繼續(xù),否則再次進入wait
        m_condVar.wait(mlock, std::bind(&Application::isDataLoaded, this));        
        std::cout << "Do Processing On loaded Data" << std::endl;
    }
};int main() {
    Application app;    
    std::thread thread_1(&Application::mainTask, &app);    
    std::thread thread_2(&Application::loadData, &app);
    thread_2.join();
    thread_1.join();    return 0;
}

“C++11多線程編程的方法”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!


標題名稱:C++11多線程編程的方法
文章位置:http://weahome.cn/article/jodcie.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部