then_deferred_pass.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  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. // <boost/thread/future.hpp>
  6. // class future<R>
  7. // template<typename F>
  8. // auto then(F&& func) -> future<decltype(func(*this))>;
  9. #define BOOST_THREAD_VERSION 4
  10. //#define BOOST_THREAD_USES_LOG
  11. #define BOOST_THREAD_USES_LOG_THREAD_ID
  12. #include <boost/thread/detail/log.hpp>
  13. #include <boost/thread/future.hpp>
  14. #include <boost/detail/lightweight_test.hpp>
  15. #include <cassert>
  16. #if defined BOOST_THREAD_PROVIDES_FUTURE_CONTINUATION
  17. #ifdef BOOST_MSVC
  18. #pragma warning(disable: 4127) // conditional expression is constant
  19. #endif
  20. int p1()
  21. {
  22. BOOST_THREAD_LOG << "p1 < " << BOOST_THREAD_END_LOG;
  23. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  24. BOOST_THREAD_LOG << "p1 >" << BOOST_THREAD_END_LOG;
  25. return 1;
  26. }
  27. int p2(boost::future<int> f)
  28. {
  29. assert(f.is_ready());
  30. BOOST_THREAD_LOG << "p2 <" << &f << BOOST_THREAD_END_LOG;
  31. BOOST_TEST(f.valid());
  32. int i = f.get();
  33. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  34. BOOST_THREAD_LOG << "p2 <" << &f << BOOST_THREAD_END_LOG;
  35. return 2 * i;
  36. }
  37. void p3(boost::future<int> f)
  38. {
  39. assert(f.is_ready());
  40. BOOST_THREAD_LOG << "p3 <" << &f << BOOST_THREAD_END_LOG;
  41. BOOST_TEST(f.valid());
  42. int i = f.get();
  43. boost::this_thread::sleep_for(boost::chrono::milliseconds(500));
  44. BOOST_THREAD_LOG << "p3 <" << &f << " " <<i << BOOST_THREAD_END_LOG;
  45. return;
  46. }
  47. int main()
  48. {
  49. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  50. {
  51. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  52. BOOST_TEST(f1.valid());
  53. boost::future<int> f2 = f1.then(boost::launch::deferred, &p2);
  54. BOOST_TEST(f2.valid());
  55. BOOST_TEST(! f1.valid());
  56. try
  57. {
  58. BOOST_TEST(f2.get()==2);
  59. }
  60. catch (std::exception& ex)
  61. {
  62. BOOST_THREAD_LOG << "ERRORRRRR "<<ex.what() << "" << BOOST_THREAD_END_LOG;
  63. BOOST_TEST(false);
  64. }
  65. catch (...)
  66. {
  67. BOOST_THREAD_LOG << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  68. BOOST_TEST(false);
  69. }
  70. }
  71. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  72. {
  73. boost::future<int> f1 = boost::async(boost::launch::async, &p1);
  74. BOOST_TEST(f1.valid());
  75. boost::future<void> f2 = f1.then(boost::launch::deferred, &p3);
  76. BOOST_TEST(f2.valid());
  77. try
  78. {
  79. f2.wait();
  80. }
  81. catch (std::exception& ex)
  82. {
  83. BOOST_THREAD_LOG << "ERRORRRRR "<<ex.what() << "" << BOOST_THREAD_END_LOG;
  84. BOOST_TEST(false);
  85. }
  86. catch (...)
  87. {
  88. BOOST_THREAD_LOG << " ERRORRRRR exception thrown" << BOOST_THREAD_END_LOG;
  89. BOOST_TEST(false);
  90. }
  91. }
  92. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  93. {
  94. boost::future<int> f2 = boost::async(p1).then(boost::launch::deferred, &p2);
  95. BOOST_TEST(f2.get()==2);
  96. }
  97. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  98. {
  99. boost::future<int> f1 = boost::async(p1);
  100. boost::future<int> f21 = f1.then(boost::launch::deferred, &p2);
  101. boost::future<int> f2= f21.then(boost::launch::deferred, &p2);
  102. BOOST_TEST(f2.get()==4);
  103. }
  104. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  105. {
  106. boost::future<int> f1 = boost::async(p1);
  107. boost::future<int> f2= f1.then(boost::launch::deferred, &p2).then(boost::launch::deferred, &p2);
  108. BOOST_TEST(f2.get()==4);
  109. }
  110. BOOST_THREAD_LOG << BOOST_THREAD_END_LOG;
  111. {
  112. boost::future<int> f2 = boost::async(p1).then(boost::launch::deferred, &p2).then(boost::launch::deferred, &p2);
  113. BOOST_TEST(f2.get()==4);
  114. }
  115. return boost::report_errors();
  116. }
  117. #else
  118. int main()
  119. {
  120. return 0;
  121. }
  122. #endif