• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //  Copyright Neil Groves 2009. Use, modification and
2 //  distribution is subject to the Boost Software License, Version
3 //  1.0. (See accompanying file LICENSE_1_0.txt or copy at
4 //  http://www.boost.org/LICENSE_1_0.txt)
5 //
6 //
7 // For more information, see http://www.boost.org/libs/range/
8 //
9 #include <boost/range/algorithm/set_algorithm.hpp>
10 
11 #include <boost/test/test_tools.hpp>
12 #include <boost/test/unit_test.hpp>
13 
14 #include <boost/assign.hpp>
15 #include <algorithm>
16 #include <functional>
17 #include <list>
18 #include <numeric>
19 #include <deque>
20 #include <vector>
21 
22 namespace boost
23 {
24     namespace
25     {
26         template<class Container1, class Iterator, class Container2>
check_result(Container1 & reference,Iterator reference_result,Container2 & test_cont,Iterator test_result)27         void check_result(
28             Container1& reference,
29             Iterator    reference_result,
30             Container2& test_cont,
31             Iterator    test_result
32             )
33         {
34             BOOST_CHECK_EQUAL(
35                 std::distance<Iterator>(reference.begin(), reference_result),
36                 std::distance<Iterator>(test_cont.begin(), test_result)
37                 );
38 
39             BOOST_CHECK_EQUAL_COLLECTIONS(
40                 reference.begin(), reference.end(),
41                 test_cont.begin(), test_cont.end()
42                 );
43         }
44 
45         template<class Container1, class Container2>
test(Container1 & cont1,Container2 & cont2)46         void test(Container1& cont1, Container2& cont2)
47         {
48             typedef BOOST_DEDUCED_TYPENAME Container1::value_type value_t;
49             typedef BOOST_DEDUCED_TYPENAME std::vector<value_t>::iterator iterator_t;
50 
51             std::vector<value_t> reference(cont1.size() + cont2.size());
52             std::vector<value_t> test_cont(reference);
53 
54             iterator_t reference_result
55                 = std::set_difference(cont1.begin(), cont1.end(),
56                                         cont2.begin(), cont2.end(),
57                                         reference.begin());
58 
59             iterator_t test_result
60                 = boost::set_difference(cont1, cont2, test_cont.begin());
61 
62             check_result(reference, reference_result,
63                          test_cont, test_result);
64 
65             test_result = boost::set_difference(
66                             boost::make_iterator_range(cont1), cont2,
67                             test_cont.begin());
68 
69             check_result(reference, reference_result,
70                          test_cont, test_result);
71 
72             test_result = boost::set_difference(
73                             cont1, boost::make_iterator_range(cont2),
74                             test_cont.begin());
75 
76             check_result(reference, reference_result,
77                          test_cont, test_result);
78 
79             test_result = boost::set_difference(
80                             boost::make_iterator_range(cont1),
81                             boost::make_iterator_range(cont2),
82                             test_cont.begin());
83 
84             check_result(reference, reference_result,
85                          test_cont, test_result);
86         }
87 
88         template<class Container, class BinaryPredicate>
sort_container(Container & cont,BinaryPredicate pred)89         void sort_container(Container& cont, BinaryPredicate pred)
90         {
91             typedef BOOST_DEDUCED_TYPENAME Container::value_type value_t;
92 
93             std::vector<value_t> temp(cont.begin(), cont.end());
94             std::sort(temp.begin(), temp.end(), pred);
95             cont.assign(temp.begin(), temp.end());
96         }
97 
98         template<class Container1,
99                  class Container2,
100                  class BinaryPredicate>
test_pred(Container1 cont1,Container2 cont2,BinaryPredicate pred)101         void test_pred(Container1 cont1, Container2 cont2,
102                        BinaryPredicate pred)
103         {
104             typedef BOOST_DEDUCED_TYPENAME Container1::value_type value_t;
105             typedef BOOST_DEDUCED_TYPENAME std::vector<value_t>::iterator iterator_t;
106 
107             sort_container(cont1, pred);
108             sort_container(cont2, pred);
109 
110             std::vector<value_t> reference(cont1.size() + cont2.size());
111             std::vector<value_t> test_cont(reference);
112 
113             iterator_t reference_result
114                 = std::set_difference(cont1.begin(), cont1.end(),
115                                         cont2.begin(), cont2.end(),
116                                         reference.begin(),
117                                         pred);
118 
119             iterator_t test_result
120                 = boost::set_difference(cont1, cont2, test_cont.begin(), pred);
121 
122             check_result(reference, reference_result,
123                          test_cont, test_result);
124 
125             test_result = boost::set_difference(
126                             boost::make_iterator_range(cont1), cont2,
127                             test_cont.begin(), pred);
128 
129             check_result(reference, reference_result,
130                          test_cont, test_result);
131 
132             test_result = boost::set_difference(
133                             cont1, boost::make_iterator_range(cont2),
134                             test_cont.begin(), pred);
135 
136             check_result(reference, reference_result,
137                          test_cont, test_result);
138 
139             test_result = boost::set_difference(
140                             boost::make_iterator_range(cont1),
141                             boost::make_iterator_range(cont2),
142                             test_cont.begin(), pred);
143 
144             check_result(reference, reference_result,
145                          test_cont, test_result);
146         }
147 
148         template<class Container1, class Container2>
test_set_difference_impl(Container1 & cont1,Container2 & cont2)149         void test_set_difference_impl(
150             Container1& cont1,
151             Container2& cont2
152             )
153         {
154             test(cont1, cont2);
155             test_pred(cont1, cont2, std::less<int>());
156             test_pred(cont1, cont2, std::greater<int>());
157         }
158 
159         template<class Container1, class Container2>
test_set_difference_impl()160         void test_set_difference_impl()
161         {
162             using namespace boost::assign;
163 
164             Container1 cont1;
165             Container2 cont2;
166 
167             test_set_difference_impl(cont1, cont2);
168 
169             cont1.clear();
170             cont2.clear();
171             cont1 += 1;
172             test_set_difference_impl(cont1, cont2);
173 
174             cont1.clear();
175             cont2.clear();
176             cont2 += 1;
177             test_set_difference_impl(cont1, cont2);
178 
179             cont1.clear();
180             cont2.clear();
181             cont1 += 1,2,3,4,5,6,7,8,9;
182             cont2 += 2,3,4;
183             test_set_difference_impl(cont1, cont2);
184 
185             cont1.clear();
186             cont2.clear();
187             cont1 += 2,3,4;
188             cont2 += 1,2,3,4,5,6,7,8,9;
189             test_set_difference_impl(cont1, cont2);
190         }
191 
test_set_difference()192         void test_set_difference()
193         {
194             test_set_difference_impl< std::vector<int>, std::vector<int> >();
195             test_set_difference_impl< std::list<int>, std::list<int> >();
196             test_set_difference_impl< std::deque<int>, std::deque<int> >();
197             test_set_difference_impl< std::vector<int>, std::list<int> >();
198             test_set_difference_impl< std::list<int>, std::vector<int> >();
199         }
200     }
201 }
202 
203 
204 boost::unit_test::test_suite*
init_unit_test_suite(int argc,char * argv[])205 init_unit_test_suite(int argc, char* argv[])
206 {
207     boost::unit_test::test_suite* test
208         = BOOST_TEST_SUITE( "RangeTestSuite.algorithm.set_difference" );
209 
210     test->add( BOOST_TEST_CASE( &boost::test_set_difference ) );
211 
212     return test;
213 }
214