C++多線程獲取返回值方法詳解
在許多時候,我們會有這樣的需求——即我們想要得到線程返回的值。但是在C++11 多線程中我們注意到,std::thread對象會忽略頂層函數(shù)的返回值。
那問題來了,我們要怎么獲得線程的返回值呢?
我們通過一個例子來說明如何實(shí)現(xiàn)這個需求。用多個線程計(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) //用三個線程,一個線程計(jì)算a+b,另一個線程計(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是封裝好的兩個類模板,這兩個類需要配合使用,他們的頭文件是#include<future>
std::future,它表示存儲著一個未來會被初始化的變量。這個變量可以通過std::future提供的成員函數(shù)std::future::get()來得到。如果在這個變量被賦值之前就有別的線程試圖通過std::future::get()獲取這個變量,那么這個線程將會被阻塞到這個變量可以獲取為止。
std::promise同樣也是一個類模板,這個對象承諾在未來一定會初始化一個變量(這個變量也就是std::future中的變量)。
每一個std::promise對象都有一個與之關(guān)聯(lián)的std::future對象。當(dāng)std::promise設(shè)置值的時候,這個值就會賦給std::future中的對象了。
#include<iostream> #include<thread> #include<mutex> #include<atomic> #include<future> //std::future std::promise #include<utility> //std::ref模板傳參的時候使用 void func2(int x, int y,std::promise<int> &promiseObj) { promiseObj.set_value(x+y); } int main() { //計(jì)算(a+b)/(x+y) //用三個線程,一個線程計(jì)算a+b,另一個線程計(jì)算x+y int a, b, x, y; a = 10, b = 8, x = 2, y = 4; int sum1, sum2; //聲明一個類 std::promise<int> promiseObj; //將future和promise關(guān)聯(lián) std::future<int> futureObj = promiseObj.get_future(); //模板傳參的時候使用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; }
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
淺析VSCode tasks.json中的各種替換變量的意思 ${workspaceFolder} ${file} ${
這篇文章主要介紹了關(guān)于VSCode tasks.json中的各種替換變量的意思 ${workspaceFolder} ${file} ${fileBasename} ${fileDirname}等,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有參考借鑒價值,需要的朋友可以參考下2020-03-03Qt實(shí)現(xiàn)驗(yàn)證碼相關(guān)功能的代碼示例
驗(yàn)證碼的原理基于人類視覺和計(jì)算機(jī)視覺的差異性,通過給用戶顯示一些難以被機(jī)器識別的圖形或文字,讓用戶進(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),文中的示例對于鏈表結(jié)點(diǎn)的操作起到了很好的說明作用,需要的朋友可以參考下2016-03-03C++實(shí)現(xiàn)json形式的Socket傳輸圖片
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)json形式的Socket傳輸圖片,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-03-03VS2022永久配置OpenCV開發(fā)環(huán)境的實(shí)現(xiàn)
本文主要介紹了VS2022永久配置OpenCV開發(fā)環(huán)境的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02