C++11獲取線程返回值的實(shí)現(xiàn)代碼
C++11 std::future and std::promise
在許多時(shí)候,我們會(huì)有這樣的需求——即我們想要得到線程返回的值。
但是在C++11 多線程中我們注意到,std::thread對(duì)象會(huì)忽略頂層函數(shù)的返回值。
那問題來了,我們要怎么獲得線程的返回值呢?
我們通過一個(gè)例子來說明如何實(shí)現(xiàn)這個(gè)需求。
假設(shè)我們的app會(huì)創(chuàng)建一個(gè)線程來壓縮一個(gè)文件夾,該線程在壓縮完文件夾后會(huì)返回壓縮文件 *.zip 和這個(gè)zip文件的大小,我們現(xiàn)在就想獲得這個(gè)線程的返回值。
有兩種方法可以實(shí)現(xiàn)這個(gè)需求:
1. 傳統(tǒng)的方法:在線程間共享指針
傳遞一個(gè)指針給壓縮文件的線程,表示壓縮文件的線程將會(huì)把值寫入指針指向的內(nèi)存空間。此時(shí)主線程將用條件變量等待值被寫入,當(dāng)壓縮文件線程把值寫入指針指定的內(nèi)存后,將喚醒(signal)條件變量,然后主線程將被喚醒,然后從指針指向的內(nèi)存中獲取返回值。
為了實(shí)現(xiàn)獲取一個(gè)返回值的需求,使用傳統(tǒng)的方法,我們需要條件變量(condition variable), 互斥量(mutex),和指針三個(gè)對(duì)象。
如果假設(shè),我們需要獲得壓縮線程里三個(gè)返回值,情況會(huì)變得更加復(fù)雜。
那std::future就是來簡化這個(gè)編程過程的
2. C++11的方法:使用std::future和std::promise
人如其名,std::future
和std::promise
對(duì)象就和他們的名字一樣。這兩個(gè)類在獲取程序返回值的時(shí)候需要配合使用
std::future,是一個(gè)類模板,它存儲(chǔ)著一個(gè)未來的值。
那問題來了,未來的值是什么鬼?
實(shí)際上一個(gè)std::future對(duì)象里存儲(chǔ)著一個(gè)在未來會(huì)被賦值的變量,這個(gè)變量可以通過std::future提供的成員函數(shù)std::future::get()來得到。如果在這個(gè)變量被賦值之前就有別的線程試圖通過std::future::get()獲取這個(gè)變量,那么這個(gè)線程將會(huì)被阻塞到這個(gè)變量可以獲取為止。
std::promise
同樣也是一個(gè)類模板,它的對(duì)象承諾會(huì)在未來設(shè)置變量(這個(gè)變量也就是std::future中的變量)。每一個(gè)std::promise對(duì)象都有一個(gè)與之關(guān)聯(lián)的std::future對(duì)象。當(dāng)std::promise設(shè)置值的時(shí)候,這個(gè)值就會(huì)賦給std::future中的對(duì)象了。
我們一步一步來看一下如何做
在主線程中創(chuàng)建std::promise對(duì)象
std::promise<int> promiseObj;
上面定義的promise對(duì)象還沒有任何關(guān)聯(lián)的值。但是它承諾某個(gè)線程將會(huì)設(shè)置與其關(guān)聯(lián)的值,并且,當(dāng)值被設(shè)置以后,可以通過與promise關(guān)聯(lián)的std::future對(duì)象來獲取該值。
假設(shè)我們的主線程將創(chuàng)建的std::promise對(duì)象傳遞給了壓縮線程,那主線程要怎么知道壓縮線程已經(jīng)設(shè)置好了值呢?
答案就是使用 std::future對(duì)象
// main thread std::future<int> futureObj = promiseObj.get_future(); int val = futureObj.get(); // compression thread promiseObj.set_value(45);
在compression thread未執(zhí)行set_value()時(shí),如果主線程調(diào)用了futureObj.get(),那么主線程將會(huì)被阻塞。
看整個(gè)時(shí)序圖:
最終代碼
#include<iostream> //std::cout std::endl #include<thread> //std::thread #include<future> //std::future std::promise #include<utility> //std::ref #include<chrono> //std::chrono::seconds void initiazer(std::promise<int> &promiseObj){ std::cout << "Inside thread: " << std::this_thread::get_id() << std::endl; std::this_thread::sleep_for(std::chrono::seconds(1)); promiseObj.set_value(35); } int main(){ std::promise<int> promiseObj; std::future<int> futureObj = promiseObj.get_future(); std::thread th(initiazer, std::ref(promiseObj)); std::cout << futureObj.get() << std::endl; th.join(); return 0; }
本文參考于C++11 Multithreading – Part 8: std::future , std::promise and Returning values from Thread,并做了適當(dāng)修改
相關(guān)文章
對(duì)C++ string append方法的常用用法詳解
今天小編就為大家分享一篇對(duì)C++ string append方法的常用用法詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-06-06C++結(jié)構(gòu)體數(shù)組詳細(xì)解析
定義結(jié)構(gòu)體數(shù)組和定義結(jié)構(gòu)體變量類似,定義結(jié)構(gòu)體數(shù)組時(shí)只需聲明其為數(shù)組即可2013-10-10C++設(shè)計(jì)模式之策略模式(Strategy)
這篇文章主要為大家詳細(xì)介紹了C++設(shè)計(jì)模式之策略模式Strategy ,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-04-04Opencv基于文字檢測去圖片水印的實(shí)現(xiàn)示例
去水印是個(gè)麻煩事,本文就來介紹一種方法Opencv基于文字檢測去圖片水印的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09C++實(shí)現(xiàn)中綴表達(dá)式轉(zhuǎn)化為后綴表達(dá)式詳解
這篇文章主要為大家詳細(xì)介紹了如何利用C++解決實(shí)現(xiàn)中綴表達(dá)式轉(zhuǎn)換為后綴表達(dá)式的問題,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03使用QGraphicsView實(shí)現(xiàn)氣泡聊天窗口+排雷功能
這篇文章主要介紹了使用QGraphicsView實(shí)現(xiàn)氣泡聊天窗口+排雷,重點(diǎn)給大家介紹使用QWebEngineView控件內(nèi)嵌html+CSS的實(shí)現(xiàn)方式,需要的朋友可以參考下2022-04-04C++?opencv利用grabCut算法實(shí)現(xiàn)摳圖示例
這篇文章主要為大家介紹了C++?opencv利用grabCut算法實(shí)現(xiàn)摳圖的代碼示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05C語言中進(jìn)行大小寫字母轉(zhuǎn)化的示例代碼
C語言標(biāo)準(zhǔn)庫中提供了用于大小寫轉(zhuǎn)換的函數(shù),使得這一操作變得簡單而高效,本文將詳細(xì)介紹如何在C語言中進(jìn)行大小寫字母的轉(zhuǎn)換,包括相關(guān)的函數(shù)和示例代碼,需要的朋友可以參考下2024-03-03