erase_equiv_tests.cpp

来自「Boost provides free peer-reviewed portab」· C++ 代码 · 共 186 行

CPP
186
字号
// Copyright 2006-2008 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 <boost/unordered_map.hpp>#include "../helpers/test.hpp"#include "../helpers/list.hpp"#include <set>#include <iostream>#include <iterator>#include <boost/next_prior.hpp>#include "../objects/test.hpp"struct write_pair_type{    template <class X1, class X2>    void operator()(std::pair<X1, X2> const& x) const    {        std::cout<<"("<<x.first<<","<<x.second<<")";    }} write_pair;template <class Container>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<int, int,    collision_hash, std::equal_to<int>,    test::allocator<std::pair<int const, int> > > collide_map;typedef boost::unordered_multimap<int, int,    collision2_hash, std::equal_to<int>,    test::allocator<std::pair<int const, int> > > collide_map2;typedef collide_map::value_type collide_value;typedef test::list<collide_value> collide_list;UNORDERED_AUTO_TEST(empty_range_tests){    collide_map x;    x.erase(x.begin(), x.end());    x.erase(x.begin(), x.begin());    x.erase(x.end(), x.end());}UNORDERED_AUTO_TEST(single_item_tests){    collide_list init;    init.push_back(collide_value(1,1));    collide_map x(init.begin(), init.end());    x.erase(x.begin(), x.begin());    BOOST_CHECK(x.count(1) == 1 && x.size() == 1);    x.erase(x.end(), x.end());    BOOST_CHECK(x.count(1) == 1 && x.size() == 1);    x.erase(x.begin(), x.end());    BOOST_CHECK(x.count(1) == 0 && x.size() == 0);}UNORDERED_AUTO_TEST(two_equivalent_item_tests){    collide_list init;    init.push_back(collide_value(1,1));    init.push_back(collide_value(1,2));    {        collide_map x(init.begin(), init.end());        x.erase(x.begin(), x.end());        BOOST_CHECK(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_CHECK(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_CHECK(x.count(1) == 1 && x.size() == 1 &&                x.begin()->first == 1 && x.begin()->second == value);    }}// More automated tests...template<class Range1, class Range2>bool compare(Range1 const& x, Range2 const& y){    collide_list a(x.begin(), x.end());    collide_list b(y.begin(), y.end());    a.sort();    b.sort();    return a == b;}template <class Container>bool general_erase_range_test(Container& x, int start, int end){    collide_list l(x.begin(), x.end());    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 <class Container>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);            collide_list init(y.begin(), y.end());            if(!general_erase_range_test(y, position, position + length)) {                BOOST_ERROR("general_erase_range_test failed.");                std::cout<<"Erase: ["<<position<<","<<position + length<<")\n";                write_container(init);                write_container(y);            }        }    }}template <class Container>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(collide_value(i, j));        }    }    std::cout<<"Values: "<<values<<", Duplicates: "<<duplicates<<"\n";    erase_subrange_tests(y);}template <class Container>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);        }    }}UNORDERED_AUTO_TEST(exhaustive_collide_tests) {    std::cout<<"exhaustive_collide_tests:\n";    collide_map m;    exhaustive_erase_tests((collide_map*) 0, 4, 4);    std::cout<<"\n";}UNORDERED_AUTO_TEST(exhaustive_collide2_tests){    std::cout<<"exhaustive_collide2_tests:\n";    exhaustive_erase_tests((collide_map2*) 0, 8, 4);    std::cout<<"\n";}RUN_TESTS()

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?