future_when_all.cpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. // Copyright (C) 2012-2013 Vicente Botet
  2. //
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  5. #include <boost/config.hpp>
  6. #if ! defined BOOST_NO_CXX11_DECLTYPE
  7. #define BOOST_RESULT_OF_USE_DECLTYPE
  8. #endif
  9. #define BOOST_THREAD_VERSION 4
  10. //#define BOOST_THREAD_USES_LOG
  11. #define BOOST_THREAD_USES_LOG_THREAD_ID
  12. #include <boost/thread/future.hpp>
  13. #include <boost/thread/csbl/vector.hpp>
  14. #include <boost/assert.hpp>
  15. #include <boost/thread/detail/log.hpp>
  16. #include <string>
  17. #if defined BOOST_THREAD_PROVIDES_FUTURE_WHEN_ALL_WHEN_ANY
  18. #ifdef BOOST_MSVC
  19. #pragma warning(disable: 4127) // conditional expression is constant
  20. #endif
  21. int p1()
  22. {
  23. BOOST_THREAD_LOG
  24. << "P1" << BOOST_THREAD_END_LOG;
  25. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  26. return 123;
  27. }
  28. int p1b()
  29. {
  30. BOOST_THREAD_LOG
  31. << "P1b" << BOOST_THREAD_END_LOG;
  32. boost::this_thread::sleep_for(boost::chrono::seconds(1));
  33. return 321;
  34. }
  35. int p2(boost::future<int> f)
  36. {
  37. BOOST_THREAD_LOG
  38. << " P2 " << BOOST_THREAD_END_LOG;
  39. try
  40. {
  41. return 2 * f.get();
  42. }
  43. catch (std::exception& ex)
  44. {
  45. BOOST_THREAD_LOG
  46. << "ERRORRRRR " << ex.what() << "" << BOOST_THREAD_END_LOG;
  47. BOOST_ASSERT(false);
  48. }
  49. catch (...)
  50. {
  51. BOOST_THREAD_LOG
  52. << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  53. BOOST_ASSERT(false);
  54. }
  55. BOOST_THREAD_LOG
  56. << "P2>" << BOOST_THREAD_END_LOG;
  57. return 0;
  58. }
  59. int p2s(boost::shared_future<int> f)
  60. {
  61. BOOST_THREAD_LOG
  62. << "<P2" << BOOST_THREAD_END_LOG;
  63. try
  64. {
  65. return 2 * f.get();
  66. }
  67. catch (std::exception& ex)
  68. {
  69. BOOST_THREAD_LOG
  70. << "ERRORRRRR " << ex.what() << "" << BOOST_THREAD_END_LOG;
  71. BOOST_ASSERT(false);
  72. }
  73. catch (...)
  74. {
  75. BOOST_THREAD_LOG
  76. << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  77. BOOST_ASSERT(false);
  78. }
  79. BOOST_THREAD_LOG
  80. << "P2>" << BOOST_THREAD_END_LOG;
  81. return 0;
  82. }
  83. int main()
  84. {
  85. BOOST_THREAD_LOG
  86. << "<MAIN" << BOOST_THREAD_END_LOG;
  87. {
  88. try
  89. {
  90. {
  91. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  92. boost::future<boost::csbl::tuple<> > all0 = boost::when_all();
  93. BOOST_THREAD_LOG
  94. << BOOST_THREAD_END_LOG;
  95. }
  96. {
  97. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  98. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  99. boost::future<boost::csbl::tuple<boost::future<int> > > all = boost::when_all(boost::move(f1));
  100. boost::csbl::tuple<boost::future<int> > res = all.get();
  101. BOOST_THREAD_LOG
  102. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  103. }
  104. #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
  105. {
  106. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  107. boost::future<int> f1 = boost::async(boost::launch::deferred, &p1);
  108. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  109. boost::future<boost::csbl::tuple<boost::future<int> > > all = boost::when_all(boost::move(f1));
  110. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  111. boost::csbl::tuple<boost::future<int> > res = all.get();
  112. BOOST_THREAD_LOG
  113. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  114. }
  115. #endif
  116. {
  117. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  118. boost::future<int> f1 = boost::make_ready_future(1);
  119. boost::future<boost::csbl::tuple<boost::future<int> > > all = boost::when_all(boost::move(f1));
  120. boost::csbl::tuple<boost::future<int> > res = all.get();
  121. BOOST_THREAD_LOG
  122. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  123. }
  124. {
  125. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  126. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  127. boost::future<int> f2 = boost::async(boost::launch::async, &p1b);
  128. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_all(boost::move(f1), boost::move(f2));
  129. //(void) all.wait();
  130. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  131. BOOST_THREAD_LOG
  132. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  133. BOOST_THREAD_LOG
  134. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  135. }
  136. {
  137. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  138. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  139. boost::future<std::string> f2 = boost::make_ready_future(std::string("nnnnnnn"));;
  140. boost::future<boost::csbl::tuple<boost::future<int>, boost::future<std::string> > > all = boost::when_all(boost::move(f1), boost::move(f2));
  141. //(void) all.wait();
  142. boost::csbl::tuple<boost::future<int>, boost::future<std::string> > res = all.get();
  143. BOOST_THREAD_LOG
  144. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  145. BOOST_THREAD_LOG
  146. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  147. }
  148. {
  149. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  150. boost::csbl::vector<boost::future<int> > v;
  151. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  152. v.push_back(boost::async(boost::launch::async, &p1));
  153. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  154. v.push_back(boost::async(boost::launch::async, &p1b));
  155. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  156. boost::future<boost::csbl::vector<boost::future<int> > > all = boost::when_all(v.begin(), v.end());
  157. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  158. boost::csbl::vector<boost::future<int> > res = all.get();
  159. BOOST_THREAD_LOG
  160. << res[0].get() <<" " << BOOST_THREAD_END_LOG;
  161. BOOST_THREAD_LOG
  162. << res[1].get() <<" " << BOOST_THREAD_END_LOG;
  163. }
  164. }
  165. catch (std::exception& ex)
  166. {
  167. BOOST_THREAD_LOG
  168. << "ERRORRRRR " << ex.what() << "" << BOOST_THREAD_END_LOG;
  169. return 1;
  170. }
  171. catch (...)
  172. {
  173. BOOST_THREAD_LOG
  174. << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  175. return 2;
  176. }
  177. }
  178. {
  179. try
  180. {
  181. {
  182. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  183. boost::future<boost::csbl::tuple<> > all0 = boost::when_any();
  184. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  185. }
  186. {
  187. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  188. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  189. boost::future<boost::csbl::tuple<boost::future<int> > > all = boost::when_any(boost::move(f1));
  190. boost::csbl::tuple<boost::future<int> > res = all.get();
  191. BOOST_THREAD_LOG
  192. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  193. }
  194. #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
  195. {
  196. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  197. boost::future<int> f1 = boost::async(boost::launch::deferred, &p1);
  198. boost::future<boost::csbl::tuple<boost::future<int> > > all = boost::when_any(boost::move(f1));
  199. boost::csbl::tuple<boost::future<int> > res = all.get();
  200. BOOST_THREAD_LOG
  201. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  202. }
  203. #endif
  204. {
  205. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  206. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  207. boost::future<int> f2 = boost::async(boost::launch::async, &p1b);
  208. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  209. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  210. BOOST_THREAD_LOG
  211. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  212. BOOST_THREAD_LOG
  213. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  214. }
  215. {
  216. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  217. boost::future<int> f1 = boost::make_ready_future(1);
  218. boost::future<int> f2 = boost::async(boost::launch::async, &p1b);
  219. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  220. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  221. BOOST_THREAD_LOG
  222. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  223. BOOST_THREAD_LOG
  224. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  225. }
  226. {
  227. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  228. boost::future<std::string> f1 = boost::make_ready_future(std::string("aaaa"));
  229. boost::future<int> f2 = boost::async(boost::launch::async, &p1b);
  230. boost::future<boost::csbl::tuple<boost::future<std::string>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  231. boost::csbl::tuple<boost::future<std::string>,boost::future<int> > res = all.get();
  232. BOOST_THREAD_LOG
  233. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  234. BOOST_THREAD_LOG
  235. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  236. }
  237. {
  238. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  239. boost::future<int> f2 = boost::make_ready_future(1);
  240. boost::future<int> f1 = boost::async(boost::launch::async, &p1b);
  241. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  242. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  243. BOOST_THREAD_LOG
  244. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  245. BOOST_THREAD_LOG
  246. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  247. }
  248. #if defined(BOOST_THREAD_PROVIDES_VARIADIC_THREAD)
  249. {
  250. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  251. boost::future<int> f1 = boost::async(boost::launch::deferred, &p1);
  252. boost::future<int> f2 = boost::async(boost::launch::async, &p1b);
  253. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  254. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  255. BOOST_THREAD_LOG
  256. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  257. BOOST_THREAD_LOG
  258. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  259. }
  260. {
  261. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  262. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  263. boost::future<int> f2 = boost::async(boost::launch::deferred, &p1b);
  264. boost::future<boost::csbl::tuple<boost::future<int>,boost::future<int> > > all = boost::when_any(boost::move(f1), boost::move(f2));
  265. boost::csbl::tuple<boost::future<int>,boost::future<int> > res = all.get();
  266. BOOST_THREAD_LOG
  267. << boost::csbl::get<0>(res).get() <<" " << BOOST_THREAD_END_LOG;
  268. BOOST_THREAD_LOG
  269. << boost::csbl::get<1>(res).get() <<" " << BOOST_THREAD_END_LOG;
  270. }
  271. #endif
  272. {
  273. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  274. boost::csbl::vector<boost::future<int> > v;
  275. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  276. v.push_back(boost::async(boost::launch::async, &p1));
  277. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  278. v.push_back(boost::async(boost::launch::async, &p1b));
  279. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  280. boost::future<boost::csbl::vector<boost::future<int> > > all = boost::when_any(v.begin(), v.end());
  281. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  282. boost::csbl::vector<boost::future<int> > res = all.get();
  283. BOOST_THREAD_LOG
  284. << res[0].get() <<" " << BOOST_THREAD_END_LOG;
  285. BOOST_THREAD_LOG
  286. << res[1].get() <<" " << BOOST_THREAD_END_LOG;
  287. }
  288. }
  289. catch (std::exception& ex)
  290. {
  291. BOOST_THREAD_LOG
  292. << "ERRORRRRR " << ex.what() << "" << BOOST_THREAD_END_LOG;
  293. return 1;
  294. }
  295. catch (...)
  296. {
  297. BOOST_THREAD_LOG
  298. << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  299. return 2;
  300. }
  301. }
  302. BOOST_THREAD_LOG
  303. << "MAIN>" << BOOST_THREAD_END_LOG;
  304. return 0;
  305. }
  306. #else
  307. #include <boost/thread/csbl/vector.hpp>
  308. using namespace boost;
  309. void f( boost::csbl::vector<future<int> > &//vec
  310. , BOOST_THREAD_RV_REF(future<int>) //f
  311. ) {
  312. }
  313. int main()
  314. {
  315. boost::csbl::vector<future<int> > vec;
  316. f(vec, make_ready_future(0));
  317. return 0;
  318. }
  319. #endif