C++多線程獲取返回值方法詳解
在許多時(shí)候,我們會(huì)有這樣的需求——即我們想要得到線程返回的值。但是在C++11 多線程中我們注意到,std::thread對(duì)象會(huì)忽略頂層函數(shù)的返回值。
那問(wèn)題來(lái)了,我們要怎么獲得線程的返回值呢?
我們通過(guò)一個(gè)例子來(lái)說(shuō)明如何實(shí)現(xiàn)這個(gè)需求。用多個(gè)線程計(jì)算(a+b)/ (x+y) 的值
有兩種方法,分別是
1. 傳統(tǒng)的方法:在線程間共享指針
#include<iostream> #include<thread> #include<mutex> #include<atomic> using namespace std; void func2(int x, int y,int* ans) { *ans= x + y; } int main() { //計(jì)算(a+b)/(x+y) //用三個(gè)線程,一個(gè)線程計(jì)算a+b,另一個(gè)線程計(jì)算x+y int a, b, x, y; a = 10, b = 8, x = 2, y = 4; int* sum1 = new int(0); int* sum2 = new int(0); thread t1(func2, a, b, sum1); t1.join(); thread t2(func2, x, y, sum2); t2.join(); cout << (*sum1) / (*sum2) << endl; delete sum1; delete sum2; system("pause"); return 0; }
2. C++11的方法:使用std::future和std::promise
std::future和std::promise是封裝好的兩個(gè)類模板,這兩個(gè)類需要配合使用,他們的頭文件是#include<future>
std::future,它表示存儲(chǔ)著一個(gè)未來(lái)會(huì)被初始化的變量。這個(gè)變量可以通過(guò)std::future提供的成員函數(shù)std::future::get()來(lái)得到。如果在這個(gè)變量被賦值之前就有別的線程試圖通過(guò)std::future::get()獲取這個(gè)變量,那么這個(gè)線程將會(huì)被阻塞到這個(gè)變量可以獲取為止。
std::promise同樣也是一個(gè)類模板,這個(gè)對(duì)象承諾在未來(lái)一定會(huì)初始化一個(gè)變量(這個(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ì)象了。
#include<iostream> #include<thread> #include<mutex> #include<atomic> #include<future> //std::future std::promise #include<utility> //std::ref模板傳參的時(shí)候使用 void func2(int x, int y,std::promise<int> &promiseObj) { promiseObj.set_value(x+y); } int main() { //計(jì)算(a+b)/(x+y) //用三個(gè)線程,一個(gè)線程計(jì)算a+b,另一個(gè)線程計(jì)算x+y int a, b, x, y; a = 10, b = 8, x = 2, y = 4; int sum1, sum2; //聲明一個(gè)類 std::promise<int> promiseObj; //將future和promise關(guān)聯(lián) std::future<int> futureObj = promiseObj.get_future(); //模板傳參的時(shí)候使用ref,否則傳參失敗 std::thread t1(func2, a, b, ref(promiseObj)); t1.join(); //獲取值 sum1 = futureObj.get(); std::cout << "sum1=" << sum1 << std::endl; //不能直接復(fù)用上面的future和promise std::promise<int> promiseObj2; std::future<int> futureObj2 = promiseObj2.get_future(); std::thread t2(func2, x, y, ref(promiseObj2)); t2.join(); sum2 = futureObj2.get(); std::cout << "sum2=" << sum2 << std::endl; std::cout << "sum1/sum2=" << sum1 / sum2 << std::endl; std::system("pause"); return 0; }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
淺析VSCode tasks.json中的各種替換變量的意思 ${workspaceFolder} ${file} ${
這篇文章主要介紹了關(guān)于VSCode tasks.json中的各種替換變量的意思 ${workspaceFolder} ${file} ${fileBasename} ${fileDirname}等,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03C++統(tǒng)計(jì)軟件使用時(shí)間代碼示例
這篇文章主要介紹了C++統(tǒng)計(jì)軟件使用時(shí)間的小程序,大家可以參考使用2013-11-11Qt實(shí)現(xiàn)驗(yàn)證碼相關(guān)功能的代碼示例
驗(yàn)證碼的原理基于人類視覺(jué)和計(jì)算機(jī)視覺(jué)的差異性,通過(guò)給用戶顯示一些難以被機(jī)器識(shí)別的圖形或文字,讓用戶進(jìn)行人機(jī)交互,確認(rèn)自己的身份,這樣可以有效保護(hù)網(wǎng)站安全,所以本給大家介紹了Qt實(shí)現(xiàn)驗(yàn)證碼相關(guān)功能的代碼示例,感興趣的朋友可以參考下2024-01-01深入解析C++的循環(huán)鏈表與雙向鏈表設(shè)計(jì)的API實(shí)現(xiàn)
這篇文章主要介紹了C++的循環(huán)鏈表與雙向鏈表設(shè)計(jì)的API實(shí)現(xiàn),文中的示例對(duì)于鏈表結(jié)點(diǎn)的操作起到了很好的說(shuō)明作用,需要的朋友可以參考下2016-03-03C++實(shí)現(xiàn)json形式的Socket傳輸圖片
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)json形式的Socket傳輸圖片,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03VS2022永久配置OpenCV開(kāi)發(fā)環(huán)境的實(shí)現(xiàn)
本文主要介紹了VS2022永久配置OpenCV開(kāi)發(fā)環(huán)境的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02C語(yǔ)言實(shí)現(xiàn)電話簿項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)電話簿項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12