// Copyright 2006-2007 Daniel James. // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // The code for erasing elements from containers with equivalent keys is very // hairy with several tricky edge cases - so explicitly test each one. #include #include #include #include #include #include #include #include "../objects/test.hpp" struct write_pair_type { template void operator()(std::pair const& x) const { std::cout<<"("< void write_container(Container const& x) { std::for_each(x.begin(), x.end(), write_pair); std::cout<<"\n"; } // Make everything collide - for testing erase in a single bucket. struct collision_hash { int operator()(int) const { return 0; } }; // For testing erase in 2 buckets. struct collision2_hash { int operator()(int x) const { return x & 1; } }; typedef boost::unordered_multimap, test::allocator > > collide_map; typedef boost::unordered_multimap, test::allocator > > collide_map2; typedef collide_map::value_type pair; typedef std::list list; void empty_range_tests() { collide_map x; x.erase(x.begin(), x.end()); x.erase(x.begin(), x.begin()); x.erase(x.end(), x.end()); } void single_item_tests() { list init; init.push_back(pair(1,1)); collide_map x(init.begin(), init.end()); x.erase(x.begin(), x.begin()); BOOST_TEST(x.count(1) == 1 && x.size() == 1); x.erase(x.end(), x.end()); BOOST_TEST(x.count(1) == 1 && x.size() == 1); x.erase(x.begin(), x.end()); BOOST_TEST(x.count(1) == 0 && x.size() == 0); } void two_equivalent_item_tests() { list init; init.push_back(pair(1,1)); init.push_back(pair(1,2)); { collide_map x(init.begin(), init.end()); x.erase(x.begin(), x.end()); BOOST_TEST(x.count(1) == 0 && x.size() == 0); } { collide_map x(init.begin(), init.end()); int value = boost::next(x.begin())->second; x.erase(x.begin(), boost::next(x.begin())); BOOST_TEST(x.count(1) == 1 && x.size() == 1 && x.begin()->first == 1 && x.begin()->second == value); } { collide_map x(init.begin(), init.end()); int value = x.begin()->second; x.erase(boost::next(x.begin()), x.end()); BOOST_TEST(x.count(1) == 1 && x.size() == 1 && x.begin()->first == 1 && x.begin()->second == value); } } // At this point I got bored and wrote more automated tests... template bool compare(Range1 const& x, Range2 const& y) { list a; list b; std::copy(x.begin(), x.end(), std::back_inserter(a)); std::copy(y.begin(), y.end(), std::back_inserter(b)); a.sort(); b.sort(); return a == b; } template bool general_erase_range_test(Container& x, int start, int end) { list l; std::copy(x.begin(), x.end(), std::back_inserter(l)); l.erase(boost::next(l.begin(), start), boost::next(l.begin(), end)); x.erase(boost::next(x.begin(), start), boost::next(x.begin(), end)); return compare(l, x); } template void erase_subrange_tests(Container const& x) { for(std::size_t length = 0; length < x.size(); ++length) { for(std::size_t position = 0; position < x.size() - length; ++position) { Container y(x); list init; std::copy(y.begin(), y.end(), std::back_inserter(init)); if(!general_erase_range_test(y, position, position + length)) { BOOST_ERROR("general_erase_range_test failed."); std::cout<<"Erase: ["< void x_by_y_erase_range_tests(Container*, int values, int duplicates) { Container y; for(int i = 0; i < values; ++i) { for(int j = 0; j < duplicates; ++j) { y.insert(pair(i, j)); } } std::cout<<"Values: "< void exhaustive_erase_tests(Container* x, int num_values, int num_duplicated) { for(int i = 0; i < num_values; ++i) { for(int j = 0; j < num_duplicated; ++j) { x_by_y_erase_range_tests(x, i, j); } } } void exhaustive_collide_tests() { std::cout<<"exhaustive_collide_tests:\n"; collide_map m; exhaustive_erase_tests((collide_map*) 0, 4, 4); std::cout<<"\n"; } void exhaustive_collide2_tests() { std::cout<<"exhaustive_collide2_tests:\n"; exhaustive_erase_tests((collide_map2*) 0, 8, 4); std::cout<<"\n"; } int main() { empty_range_tests(); single_item_tests(); two_equivalent_item_tests(); exhaustive_collide_tests(); exhaustive_collide2_tests(); return boost::report_errors(); }