2
// Copyright 2006-2009 Daniel James.
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)
6
#include "../helpers/prefix.hpp"
8
#include <boost/unordered_set.hpp>
9
#include <boost/unordered_map.hpp>
10
#include "../helpers/test.hpp"
11
#include <boost/next_prior.hpp>
12
#include "../objects/test.hpp"
13
#include "../helpers/random_values.hpp"
14
#include "../helpers/tracker.hpp"
15
#include "../helpers/equivalent.hpp"
16
#include "../helpers/helpers.hpp"
23
test::seed_t seed(85638);
25
template <class Container>
26
void erase_tests1(Container*,
27
test::random_generator generator = test::default_generator)
29
std::cerr<<"Erase by key.\n";
31
test::check_instances check_;
33
test::random_values<Container> v(1000, generator);
34
Container x(v.begin(), v.end());
35
for(BOOST_DEDUCED_TYPENAME test::random_values<Container>::iterator
36
it = v.begin(); it != v.end(); ++it)
38
std::size_t count = x.count(test::get_key<Container>(*it));
39
std::size_t old_size = x.size();
40
BOOST_TEST(count == x.erase(test::get_key<Container>(*it)));
41
BOOST_TEST(x.size() == old_size - count);
42
BOOST_TEST(x.count(test::get_key<Container>(*it)) == 0);
43
BOOST_TEST(x.find(test::get_key<Container>(*it)) == x.end());
47
std::cerr<<"erase(begin()).\n";
49
test::check_instances check_;
51
test::random_values<Container> v(1000, generator);
52
Container x(v.begin(), v.end());
53
std::size_t size = x.size();
54
while(size > 0 && !x.empty())
56
BOOST_DEDUCED_TYPENAME Container::key_type
57
key = test::get_key<Container>(*x.begin());
58
std::size_t count = x.count(key);
59
BOOST_DEDUCED_TYPENAME Container::iterator
60
pos = x.erase(x.begin());
62
BOOST_TEST(pos == x.begin());
63
BOOST_TEST(x.count(key) == count - 1);
64
BOOST_TEST(x.size() == size);
66
BOOST_TEST(x.empty());
69
std::cerr<<"erase(random position).\n";
71
test::check_instances check_;
73
test::random_values<Container> v(1000, generator);
74
Container x(v.begin(), v.end());
75
std::size_t size = x.size();
76
while(size > 0 && !x.empty())
79
int index = rand() % (int) x.size();
80
BOOST_DEDUCED_TYPENAME Container::const_iterator prev, pos, next;
82
prev = pos = x.begin();
85
prev = boost::next(x.begin(), index - 1);
86
pos = boost::next(prev);
88
next = boost::next(pos);
89
BOOST_DEDUCED_TYPENAME Container::key_type
90
key = test::get_key<Container>(*pos);
91
std::size_t count = x.count(key);
92
BOOST_TEST(next == x.erase(pos));
95
BOOST_TEST(index == 0 ? next == x.begin() :
96
next == boost::next(prev));
97
BOOST_TEST(x.count(key) == count - 1);
98
BOOST_TEST(x.size() == size);
100
BOOST_TEST(x.empty());
103
std::cerr<<"erase(ranges).\n";
105
test::check_instances check_;
107
test::random_values<Container> v(500, generator);
108
Container x(v.begin(), v.end());
110
std::size_t size = x.size();
112
// I'm actually stretching it a little here, as the standard says it
113
// returns 'the iterator immediately following the erase elements'
114
// and if nothing is erased, then there's nothing to follow. But I
115
// think this is the only sensible option...
116
BOOST_TEST(x.erase(x.end(), x.end()) == x.end());
117
BOOST_TEST(x.erase(x.begin(), x.begin()) == x.begin());
118
BOOST_TEST(x.size() == size);
120
BOOST_TEST(x.erase(x.begin(), x.end()) == x.end());
121
BOOST_TEST(x.empty());
122
BOOST_TEST(x.begin() == x.end());
124
BOOST_TEST(x.erase(x.begin(), x.end()) == x.begin());
127
std::cerr<<"quick_erase(begin()).\n";
129
test::check_instances check_;
131
test::random_values<Container> v(1000, generator);
132
Container x(v.begin(), v.end());
133
std::size_t size = x.size();
134
while(size > 0 && !x.empty())
136
BOOST_DEDUCED_TYPENAME Container::key_type
137
key = test::get_key<Container>(*x.begin());
138
std::size_t count = x.count(key);
139
x.quick_erase(x.begin());
141
BOOST_TEST(x.count(key) == count - 1);
142
BOOST_TEST(x.size() == size);
144
BOOST_TEST(x.empty());
147
std::cerr<<"quick_erase(random position).\n";
149
test::check_instances check_;
151
test::random_values<Container> v(1000, generator);
152
Container x(v.begin(), v.end());
153
std::size_t size = x.size();
154
while(size > 0 && !x.empty())
157
int index = rand() % (int) x.size();
158
BOOST_DEDUCED_TYPENAME Container::const_iterator prev, pos, next;
160
prev = pos = x.begin();
163
prev = boost::next(x.begin(), index - 1);
164
pos = boost::next(prev);
166
next = boost::next(pos);
167
BOOST_DEDUCED_TYPENAME Container::key_type
168
key = test::get_key<Container>(*pos);
169
std::size_t count = x.count(key);
173
BOOST_TEST(index == 0 ? next == x.begin() :
174
next == boost::next(prev));
175
BOOST_TEST(x.count(key) == count - 1);
176
BOOST_TEST(x.size() == size);
178
BOOST_TEST(x.empty());
182
std::cerr<<"clear().\n";
184
test::check_instances check_;
186
test::random_values<Container> v(500, generator);
187
Container x(v.begin(), v.end());
189
BOOST_TEST(x.empty());
190
BOOST_TEST(x.begin() == x.end());
196
boost::unordered_set<test::object,
197
test::hash, test::equal_to,
198
test::allocator<test::object> >* test_set;
199
boost::unordered_multiset<test::object,
200
test::hash, test::equal_to,
201
test::allocator<test::object> >* test_multiset;
202
boost::unordered_map<test::object, test::object,
203
test::hash, test::equal_to,
204
test::allocator<test::object> >* test_map;
205
boost::unordered_multimap<test::object, test::object,
206
test::hash, test::equal_to,
207
test::allocator<test::object> >* test_multimap;
209
using test::default_generator;
210
using test::generate_collisions;
212
UNORDERED_TEST(erase_tests1,
213
((test_set)(test_multiset)(test_map)(test_multimap))
214
((default_generator)(generate_collisions))