forked from boostorg/unordered
		
	
		
			
				
	
	
		
			310 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2021 Christian Mazakas.
 | |
| // 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)
 | |
| 
 | |
| // clang-format off
 | |
| #include "../helpers/prefix.hpp"
 | |
| #include <boost/unordered_set.hpp>
 | |
| #include <boost/unordered_map.hpp>
 | |
| #include "../helpers/postfix.hpp"
 | |
| // clang-format on
 | |
| 
 | |
| #include "../helpers/test.hpp"
 | |
| 
 | |
| #include <boost/container_hash/hash.hpp>
 | |
| 
 | |
| struct key
 | |
| {
 | |
|   int x_;
 | |
|   static int count_;
 | |
| 
 | |
|   key(int x) : x_(x) { ++count_; }
 | |
|   key(key const& k) : x_(k.x_) { ++count_; }
 | |
| };
 | |
| 
 | |
| int key::count_;
 | |
| 
 | |
| struct transparent_hasher
 | |
| {
 | |
|   typedef void is_transparent;
 | |
| 
 | |
|   std::size_t operator()(key const& k) const
 | |
|   {
 | |
|     return boost::hash<int>()(k.x_);
 | |
|   }
 | |
| 
 | |
|   std::size_t operator()(int const k) const { return boost::hash<int>()(k); }
 | |
| };
 | |
| 
 | |
| struct transparent_key_equal
 | |
| {
 | |
|   typedef void is_transparent;
 | |
| 
 | |
|   static bool was_called_;
 | |
| 
 | |
|   bool operator()(key const& k1, key const& k2) const { return k1.x_ == k2.x_; }
 | |
|   bool operator()(int const x, key const& k1) const
 | |
|   {
 | |
|     was_called_ = true;
 | |
|     return k1.x_ == x;
 | |
|   }
 | |
| };
 | |
| 
 | |
| bool transparent_key_equal::was_called_;
 | |
| 
 | |
| struct hasher
 | |
| {
 | |
|   std::size_t operator()(key const& k) const
 | |
|   {
 | |
|     return boost::hash<int>()(k.x_);
 | |
|   }
 | |
| };
 | |
| 
 | |
| struct key_equal
 | |
| {
 | |
|   static bool was_called_;
 | |
| 
 | |
|   bool operator()(key const& k1, key const& k2) const { return k1.x_ == k2.x_; }
 | |
|   bool operator()(int const x, key const& k1) const
 | |
|   {
 | |
|     was_called_ = true;
 | |
|     return k1.x_ == x;
 | |
|   }
 | |
| };
 | |
| 
 | |
| bool key_equal::was_called_;
 | |
| 
 | |
| void count_reset()
 | |
| {
 | |
|   key::count_ = 0;
 | |
|   transparent_key_equal::was_called_ = false;
 | |
|   key_equal::was_called_ = false;
 | |
| }
 | |
| 
 | |
| template <class UnorderedMap> void test_transparent_count()
 | |
| {
 | |
|   count_reset();
 | |
| 
 | |
|   UnorderedMap map;
 | |
| 
 | |
|   // initial `key(0)` expression increases the count
 | |
|   // then copying into the `unordered_map` increments the count again thus we
 | |
|   // have 2
 | |
|   //
 | |
|   map[key(0)] = 1337;
 | |
|   BOOST_TEST(key::count_ == 2);
 | |
| 
 | |
|   // now the number of `key` objects created should be a constant and never
 | |
|   // touched again
 | |
|   //
 | |
|   std::size_t count = 0;
 | |
|   count = map.count(0);
 | |
| 
 | |
|   BOOST_TEST(count == 1);
 | |
|   BOOST_TEST(key::count_ == 2);
 | |
|   BOOST_TEST(map.key_eq().was_called_);
 | |
| 
 | |
|   count = map.count(1);
 | |
| 
 | |
|   BOOST_TEST(count == 0);
 | |
|   BOOST_TEST(key::count_ == 2);
 | |
| 
 | |
|   count = map.count(key(0));
 | |
|   BOOST_TEST(count == 1);
 | |
|   BOOST_TEST(key::count_ == 3);
 | |
| }
 | |
| 
 | |
| template <class UnorderedMap> void test_non_transparent_count()
 | |
| {
 | |
|   count_reset();
 | |
| 
 | |
|   UnorderedMap map;
 | |
| 
 | |
|   // initial `key(0)` expression increases the count
 | |
|   // then copying into the `unordered_map` increments the count again thus we
 | |
|   // have 2
 | |
|   //
 | |
|   map[key(0)] = 1337;
 | |
|   BOOST_TEST(key::count_ == 2);
 | |
| 
 | |
|   // rely on the implicit constructor here to spawn a new object which
 | |
|   // increases the count
 | |
|   //
 | |
|   std::size_t count = 0;
 | |
|   count = map.count(0);
 | |
| 
 | |
|   BOOST_TEST(count == 1);
 | |
|   BOOST_TEST(key::count_ == 3);
 | |
|   BOOST_TEST(!map.key_eq().was_called_);
 | |
| 
 | |
|   count = map.count(1);
 | |
| 
 | |
|   BOOST_TEST(count == 0);
 | |
|   BOOST_TEST(key::count_ == 4);
 | |
| 
 | |
|   count = map.count(key(0));
 | |
|   BOOST_TEST(count == 1);
 | |
|   BOOST_TEST(key::count_ == 5);
 | |
| }
 | |
| 
 | |
| template <class UnorderedMap> void test_transparent_find()
 | |
| {
 | |
|   count_reset();
 | |
| 
 | |
|   typedef typename UnorderedMap::const_iterator map_iterator;
 | |
|   typedef typename UnorderedMap::value_type pair;
 | |
| 
 | |
|   UnorderedMap map;
 | |
| 
 | |
|   int n = 5;
 | |
| 
 | |
|   for (int i = 0; i < n; ++i) {
 | |
|     map[key(i)] = i;
 | |
|   }
 | |
| 
 | |
|   int const expected_key_count = 2 * n;
 | |
|   BOOST_TEST(key::count_ == expected_key_count);
 | |
| 
 | |
|   {
 | |
|     UnorderedMap& m = map;
 | |
| 
 | |
|     for (int i = 0; i < n; ++i) {
 | |
|       map_iterator pos = m.find(i);
 | |
|       BOOST_TEST(pos != m.end());
 | |
| 
 | |
|       pair const& p = *pos;
 | |
|       int const v = p.second;
 | |
| 
 | |
|       BOOST_TEST(v == i);
 | |
|     }
 | |
| 
 | |
|     BOOST_TEST(key::count_ == expected_key_count);
 | |
| 
 | |
|     map_iterator pos = m.find(1337);
 | |
|     BOOST_TEST(pos == m.end());
 | |
|     BOOST_TEST(key::count_ == expected_key_count);
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     UnorderedMap const& m = map;
 | |
| 
 | |
|     for (int i = 0; i < n; ++i) {
 | |
|       map_iterator pos = m.find(i);
 | |
|       BOOST_TEST(pos != m.end());
 | |
| 
 | |
|       pair const& p = *pos;
 | |
|       int const v = p.second;
 | |
| 
 | |
|       BOOST_TEST(v == i);
 | |
|     }
 | |
| 
 | |
|     BOOST_TEST(key::count_ == expected_key_count);
 | |
| 
 | |
|     map_iterator pos = m.find(1337);
 | |
|     BOOST_TEST(pos == m.end());
 | |
|     BOOST_TEST(key::count_ == expected_key_count);
 | |
|   }
 | |
| }
 | |
| 
 | |
| template <class UnorderedMap> void test_non_transparent_find()
 | |
| {
 | |
|   count_reset();
 | |
| 
 | |
|   typedef typename UnorderedMap::const_iterator map_iterator;
 | |
|   typedef typename UnorderedMap::value_type pair;
 | |
| 
 | |
|   UnorderedMap map;
 | |
| 
 | |
|   int n = 5;
 | |
| 
 | |
|   for (int i = 0; i < n; ++i) {
 | |
|     map[key(i)] = i;
 | |
|   }
 | |
| 
 | |
|   int key_count = 2 * n;
 | |
| 
 | |
|   BOOST_TEST(key::count_ == key_count);
 | |
| 
 | |
|   {
 | |
|     UnorderedMap& m = map;
 | |
| 
 | |
|     for (int i = 0; i < n; ++i) {
 | |
|       map_iterator pos = m.find(i);
 | |
|       BOOST_TEST(pos != m.end());
 | |
| 
 | |
|       pair const& p = *pos;
 | |
|       int const v = p.second;
 | |
| 
 | |
|       BOOST_TEST(v == i);
 | |
|     }
 | |
|     BOOST_TEST(key::count_ == n + key_count);
 | |
| 
 | |
|     map_iterator pos = m.find(1337);
 | |
|     BOOST_TEST(pos == m.end());
 | |
|     BOOST_TEST(key::count_ == 1 + n + key_count);
 | |
| 
 | |
|     key_count = key::count_;
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     UnorderedMap const& m = map;
 | |
| 
 | |
|     for (int i = 0; i < n; ++i) {
 | |
|       map_iterator pos = m.find(i);
 | |
|       BOOST_TEST(pos != m.end());
 | |
| 
 | |
|       pair const& p = *pos;
 | |
|       int const v = p.second;
 | |
| 
 | |
|       BOOST_TEST(v == i);
 | |
|     }
 | |
|     BOOST_TEST(key::count_ == n + key_count);
 | |
| 
 | |
|     map_iterator pos = m.find(1337);
 | |
|     BOOST_TEST(pos == m.end());
 | |
|     BOOST_TEST(key::count_ == 1 + n + key_count);
 | |
|   }
 | |
| }
 | |
| 
 | |
| UNORDERED_AUTO_TEST (unordered_map_transparent_count) {
 | |
|   {
 | |
|     typedef boost::unordered_map<key, int, transparent_hasher,
 | |
|       transparent_key_equal>
 | |
|       unordered_map;
 | |
| 
 | |
|     test_transparent_count<unordered_map>();
 | |
|     test_transparent_find<unordered_map>();
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     // non-transparent Hash, non-transparent KeyEqual
 | |
|     //
 | |
|     typedef boost::unordered_map<key, int, hasher, key_equal> unordered_map;
 | |
| 
 | |
|     test_non_transparent_count<unordered_map>();
 | |
|     test_non_transparent_find<unordered_map>();
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     // transparent Hash, non-transparent KeyEqual
 | |
|     //
 | |
|     typedef boost::unordered_map<key, int, transparent_hasher, key_equal>
 | |
|       unordered_map;
 | |
| 
 | |
|     test_non_transparent_count<unordered_map>();
 | |
|     test_non_transparent_find<unordered_map>();
 | |
|   }
 | |
| 
 | |
|   {
 | |
|     // non-transparent Hash, transparent KeyEqual
 | |
|     //
 | |
|     typedef boost::unordered_map<key, int, hasher, transparent_key_equal>
 | |
|       unordered_map;
 | |
| 
 | |
|     test_non_transparent_count<unordered_map>();
 | |
|     test_non_transparent_find<unordered_map>();
 | |
|   }
 | |
| }
 | |
| 
 | |
| RUN_TESTS()
 |