transform_reduce_test.cpp 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. /*
  2. Copyright (c) Marshall Clow 2013.
  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. For more information, see http://www.boost.org
  6. */
  7. #include <boost/config.hpp>
  8. #include <boost/algorithm/cxx17/transform_reduce.hpp>
  9. #include "iterator_test.hpp"
  10. #define BOOST_TEST_MAIN
  11. #include <boost/test/unit_test.hpp>
  12. namespace ba = boost::algorithm;
  13. template <class _Tp>
  14. struct identity
  15. {
  16. const _Tp& operator()(const _Tp& __x) const { return __x;}
  17. };
  18. template <class _Tp>
  19. struct twice
  20. {
  21. const _Tp operator()(const _Tp& __x) const { return 2 * __x; }
  22. };
  23. template <class Iter1, class T, class BOp, class UOp>
  24. void
  25. test_init_bop_uop(Iter1 first1, Iter1 last1, T init, BOp bOp, UOp uOp, T x)
  26. {
  27. BOOST_CHECK(ba::transform_reduce(first1, last1, init, bOp, uOp) == x);
  28. }
  29. template <class Iter>
  30. void
  31. test_init_bop_uop()
  32. {
  33. int ia[] = {1, 2, 3, 4, 5, 6};
  34. unsigned sa = sizeof(ia) / sizeof(ia[0]);
  35. test_init_bop_uop(Iter(ia), Iter(ia), 0, std::plus<int>(), identity<int>(), 0);
  36. test_init_bop_uop(Iter(ia), Iter(ia), 1, std::multiplies<int>(), identity<int>(), 1);
  37. test_init_bop_uop(Iter(ia), Iter(ia+1), 0, std::multiplies<int>(), identity<int>(), 0);
  38. test_init_bop_uop(Iter(ia), Iter(ia+1), 2, std::plus<int>(), identity<int>(), 3);
  39. test_init_bop_uop(Iter(ia), Iter(ia+2), 0, std::plus<int>(), identity<int>(), 3);
  40. test_init_bop_uop(Iter(ia), Iter(ia+2), 3, std::multiplies<int>(), identity<int>(), 6);
  41. test_init_bop_uop(Iter(ia), Iter(ia+sa), 4, std::multiplies<int>(), identity<int>(), 2880);
  42. test_init_bop_uop(Iter(ia), Iter(ia+sa), 4, std::plus<int>(), identity<int>(), 25);
  43. test_init_bop_uop(Iter(ia), Iter(ia), 0, std::plus<int>(), twice<int>(), 0);
  44. test_init_bop_uop(Iter(ia), Iter(ia), 1, std::multiplies<int>(), twice<int>(), 1);
  45. test_init_bop_uop(Iter(ia), Iter(ia+1), 0, std::multiplies<int>(), twice<int>(), 0);
  46. test_init_bop_uop(Iter(ia), Iter(ia+1), 2, std::plus<int>(), twice<int>(), 4);
  47. test_init_bop_uop(Iter(ia), Iter(ia+2), 0, std::plus<int>(), twice<int>(), 6);
  48. test_init_bop_uop(Iter(ia), Iter(ia+2), 3, std::multiplies<int>(), twice<int>(), 24);
  49. test_init_bop_uop(Iter(ia), Iter(ia+sa), 4, std::multiplies<int>(), twice<int>(), 184320); // 64 * 2880
  50. test_init_bop_uop(Iter(ia), Iter(ia+sa), 4, std::plus<int>(), twice<int>(), 46);
  51. }
  52. void test_transform_reduce_init_bop_uop()
  53. {
  54. BOOST_CHECK ( true );
  55. }
  56. template <class Iter1, class Iter2, class T, class Op1, class Op2>
  57. void
  58. test_init_bop_bop(Iter1 first1, Iter1 last1, Iter2 first2, T init, Op1 op1, Op2 op2, T x)
  59. {
  60. BOOST_CHECK(ba::transform_reduce(first1, last1, first2, init, op1, op2) == x);
  61. }
  62. template <class SIter, class UIter>
  63. void
  64. test_init_bop_bop()
  65. {
  66. int ia[] = {1, 2, 3, 4, 5, 6};
  67. unsigned int ua[] = {2, 4, 6, 8, 10,12};
  68. unsigned sa = sizeof(ia) / sizeof(ia[0]);
  69. BOOST_CHECK(sa == sizeof(ua) / sizeof(ua[0])); // just to be sure
  70. test_init_bop_bop(SIter(ia), SIter(ia), UIter(ua), 0, std::plus<int>(), std::multiplies<int>(), 0);
  71. test_init_bop_bop(UIter(ua), UIter(ua), SIter(ia), 1, std::multiplies<int>(), std::plus<int>(), 1);
  72. test_init_bop_bop(SIter(ia), SIter(ia+1), UIter(ua), 0, std::multiplies<int>(), std::plus<int>(), 0);
  73. test_init_bop_bop(UIter(ua), UIter(ua+1), SIter(ia), 2, std::plus<int>(), std::multiplies<int>(), 4);
  74. test_init_bop_bop(SIter(ia), SIter(ia+2), UIter(ua), 0, std::plus<int>(), std::multiplies<int>(), 10);
  75. test_init_bop_bop(UIter(ua), UIter(ua+2), SIter(ia), 3, std::multiplies<int>(), std::plus<int>(), 54);
  76. test_init_bop_bop(SIter(ia), SIter(ia+sa), UIter(ua), 4, std::multiplies<int>(), std::plus<int>(), 2099520);
  77. test_init_bop_bop(UIter(ua), UIter(ua+sa), SIter(ia), 4, std::plus<int>(), std::multiplies<int>(), 186);
  78. }
  79. void test_transform_reduce_init_bop_bop()
  80. {
  81. // All the iterator categories
  82. test_init_bop_bop<input_iterator <const int*>, input_iterator <const unsigned int*> >();
  83. test_init_bop_bop<input_iterator <const int*>, forward_iterator <const unsigned int*> >();
  84. test_init_bop_bop<input_iterator <const int*>, bidirectional_iterator<const unsigned int*> >();
  85. test_init_bop_bop<input_iterator <const int*>, random_access_iterator<const unsigned int*> >();
  86. test_init_bop_bop<forward_iterator <const int*>, input_iterator <const unsigned int*> >();
  87. test_init_bop_bop<forward_iterator <const int*>, forward_iterator <const unsigned int*> >();
  88. test_init_bop_bop<forward_iterator <const int*>, bidirectional_iterator<const unsigned int*> >();
  89. test_init_bop_bop<forward_iterator <const int*>, random_access_iterator<const unsigned int*> >();
  90. test_init_bop_bop<bidirectional_iterator<const int*>, input_iterator <const unsigned int*> >();
  91. test_init_bop_bop<bidirectional_iterator<const int*>, forward_iterator <const unsigned int*> >();
  92. test_init_bop_bop<bidirectional_iterator<const int*>, bidirectional_iterator<const unsigned int*> >();
  93. test_init_bop_bop<bidirectional_iterator<const int*>, random_access_iterator<const unsigned int*> >();
  94. test_init_bop_bop<random_access_iterator<const int*>, input_iterator <const unsigned int*> >();
  95. test_init_bop_bop<random_access_iterator<const int*>, forward_iterator <const unsigned int*> >();
  96. test_init_bop_bop<random_access_iterator<const int*>, bidirectional_iterator<const unsigned int*> >();
  97. test_init_bop_bop<random_access_iterator<const int*>, random_access_iterator<const unsigned int*> >();
  98. // just plain pointers (const vs. non-const, too)
  99. test_init_bop_bop<const int*, const unsigned int *>();
  100. test_init_bop_bop<const int*, unsigned int *>();
  101. test_init_bop_bop< int*, const unsigned int *>();
  102. test_init_bop_bop< int*, unsigned int *>();
  103. }
  104. template <class Iter1, class Iter2, class T>
  105. void
  106. test_init(Iter1 first1, Iter1 last1, Iter2 first2, T init, T x)
  107. {
  108. BOOST_CHECK(ba::transform_reduce(first1, last1, first2, init) == x);
  109. }
  110. template <class SIter, class UIter>
  111. void
  112. test_init()
  113. {
  114. int ia[] = {1, 2, 3, 4, 5, 6};
  115. unsigned int ua[] = {2, 4, 6, 8, 10,12};
  116. unsigned sa = sizeof(ia) / sizeof(ia[0]);
  117. BOOST_CHECK(sa == sizeof(ua) / sizeof(ua[0])); // just to be sure
  118. test_init(SIter(ia), SIter(ia), UIter(ua), 0, 0);
  119. test_init(UIter(ua), UIter(ua), SIter(ia), 1, 1);
  120. test_init(SIter(ia), SIter(ia+1), UIter(ua), 0, 2);
  121. test_init(UIter(ua), UIter(ua+1), SIter(ia), 2, 4);
  122. test_init(SIter(ia), SIter(ia+2), UIter(ua), 0, 10);
  123. test_init(UIter(ua), UIter(ua+2), SIter(ia), 3, 13);
  124. test_init(SIter(ia), SIter(ia+sa), UIter(ua), 0, 182);
  125. test_init(UIter(ua), UIter(ua+sa), SIter(ia), 4, 186);
  126. }
  127. void test_transform_reduce_init()
  128. {
  129. // All the iterator categories
  130. test_init<input_iterator <const int*>, input_iterator <const unsigned int*> >();
  131. test_init<input_iterator <const int*>, forward_iterator <const unsigned int*> >();
  132. test_init<input_iterator <const int*>, bidirectional_iterator<const unsigned int*> >();
  133. test_init<input_iterator <const int*>, random_access_iterator<const unsigned int*> >();
  134. test_init<forward_iterator <const int*>, input_iterator <const unsigned int*> >();
  135. test_init<forward_iterator <const int*>, forward_iterator <const unsigned int*> >();
  136. test_init<forward_iterator <const int*>, bidirectional_iterator<const unsigned int*> >();
  137. test_init<forward_iterator <const int*>, random_access_iterator<const unsigned int*> >();
  138. test_init<bidirectional_iterator<const int*>, input_iterator <const unsigned int*> >();
  139. test_init<bidirectional_iterator<const int*>, forward_iterator <const unsigned int*> >();
  140. test_init<bidirectional_iterator<const int*>, bidirectional_iterator<const unsigned int*> >();
  141. test_init<bidirectional_iterator<const int*>, random_access_iterator<const unsigned int*> >();
  142. test_init<random_access_iterator<const int*>, input_iterator <const unsigned int*> >();
  143. test_init<random_access_iterator<const int*>, forward_iterator <const unsigned int*> >();
  144. test_init<random_access_iterator<const int*>, bidirectional_iterator<const unsigned int*> >();
  145. test_init<random_access_iterator<const int*>, random_access_iterator<const unsigned int*> >();
  146. // just plain pointers (const vs. non-const, too)
  147. test_init<const int*, const unsigned int *>();
  148. test_init<const int*, unsigned int *>();
  149. test_init< int*, const unsigned int *>();
  150. test_init< int*, unsigned int *>();
  151. }
  152. BOOST_AUTO_TEST_CASE( test_main )
  153. {
  154. test_transform_reduce_init();
  155. test_transform_reduce_init_bop_uop();
  156. test_transform_reduce_init_bop_bop();
  157. }