Combine hash_structure and hash_table_manager.

[SVN r55990]
This commit is contained in:
Daniel James
2009-09-03 07:36:21 +00:00
parent 1743ed118d
commit 7fe53ef5a3
6 changed files with 241 additions and 331 deletions

View File

@ -152,76 +152,13 @@ namespace boost { namespace unordered_detail {
template <class NodeBase, class ValueType>
class hash_node : public NodeBase, public value_base<ValueType>
{
public:
typedef ValueType value_type;
typedef BOOST_DEDUCED_TYPENAME NodeBase::node_ptr node_ptr;
public:
static value_type& get_value(node_ptr p) { return static_cast<hash_node&>(*p).value(); }
};
template <class A, class G>
struct hash_structure
{
typedef BOOST_DEDUCED_TYPENAME
G::BOOST_NESTED_TEMPLATE base<A>::type
node_base;
typedef BOOST_DEDUCED_TYPENAME node_base::bucket_allocator bucket_allocator;
typedef BOOST_DEDUCED_TYPENAME node_base::bucket_ptr bucket_ptr;
typedef BOOST_DEDUCED_TYPENAME node_base::node_ptr node_ptr;
// The actual data structure
bucket_ptr buckets_;
bucket_ptr cached_begin_bucket_;
std::size_t size_;
std::size_t bucket_count_;
// Constructor
hash_structure() : buckets_(), cached_begin_bucket_(), size_() {}
void swap(hash_structure& other);
// Buckets
std::size_t bucket_count() const;
std::size_t bucket_from_hash(std::size_t hashed) const;
bucket_ptr bucket_ptr_from_hash(std::size_t hashed) const;
bucket_ptr buckets_begin() const;
bucket_ptr buckets_end() const;
std::size_t bucket_size(std::size_t index) const;
// Link a node
void link_node(node_ptr n, node_ptr position);
void link_node_in_bucket(node_ptr n, bucket_ptr bucket);
void unlink_node(bucket_ptr bucket, node_ptr pos);
void unlink_nodes(bucket_ptr bucket, node_ptr begin, node_ptr end);
void unlink_nodes(bucket_ptr bucket, node_ptr end);
std::size_t unlink_group(node_ptr* pos);
void link_group(node_ptr n, bucket_ptr bucket, std::size_t count);
bucket_ptr get_bucket(std::size_t n) const;
node_ptr bucket_begin(std::size_t n) const;
node_ptr bucket_end(std::size_t) const;
// recompute_begin_bucket
//
// After an erase cached_begin_bucket_ might be left pointing to
// an empty bucket, so this is called to update it
//
// no throw
void recompute_begin_bucket(bucket_ptr b);
// This is called when a range has been erased
//
// no throw
void recompute_begin_bucket(bucket_ptr b1, bucket_ptr b2);
// no throw
float load_factor() const;
};
// Iterator Base
template <class BucketPtr>
@ -256,22 +193,22 @@ namespace boost { namespace unordered_detail {
// methods (other than getters and setters).
template <class A, class G>
struct hash_table_manager :
hash_structure<A, G>
struct hash_table_manager
{
// Types
typedef hash_bucket<A> bucket;
typedef hash_structure<A, G> structure;
typedef BOOST_DEDUCED_TYPENAME structure::bucket_allocator bucket_allocator;
typedef BOOST_DEDUCED_TYPENAME structure::node_base node_base;
typedef BOOST_DEDUCED_TYPENAME structure::bucket_ptr bucket_ptr;
typedef BOOST_DEDUCED_TYPENAME structure::node_ptr node_ptr;
typedef A value_allocator;
typedef BOOST_DEDUCED_TYPENAME A::value_type value_type;
typedef hash_bucket<A> bucket;
typedef BOOST_DEDUCED_TYPENAME G::BOOST_NESTED_TEMPLATE base<A>::type
node_base;
typedef hash_node<node_base, value_type> node;
typedef BOOST_DEDUCED_TYPENAME node::bucket_allocator bucket_allocator;
typedef BOOST_DEDUCED_TYPENAME node::bucket_ptr bucket_ptr;
typedef BOOST_DEDUCED_TYPENAME node::node_ptr node_ptr;
typedef BOOST_DEDUCED_TYPENAME rebind_wrap<value_allocator, node>::type node_allocator;
typedef BOOST_DEDUCED_TYPENAME node_allocator::pointer real_node_ptr;
@ -279,6 +216,10 @@ namespace boost { namespace unordered_detail {
// Members
bucket_ptr buckets_;
bucket_ptr cached_begin_bucket_;
std::size_t size_;
std::size_t bucket_count_;
boost::compressed_pair<bucket_allocator, node_allocator> allocators_;
// Data access
@ -306,11 +247,21 @@ namespace boost { namespace unordered_detail {
~hash_table_manager();
// no throw
void swap(hash_table_manager& other);
void move(hash_table_manager& other);
// Methods
// Buckets
void create_buckets(std::size_t bucket_count);
std::size_t bucket_count() const;
std::size_t bucket_from_hash(std::size_t hashed) const;
bucket_ptr bucket_ptr_from_hash(std::size_t hashed) const;
bucket_ptr buckets_begin() const;
bucket_ptr buckets_end() const;
std::size_t bucket_size(std::size_t index) const;
bucket_ptr get_bucket(std::size_t n) const;
node_ptr bucket_begin(std::size_t n) const;
node_ptr bucket_end(std::size_t) const;
// Alloc/Dealloc
@ -331,6 +282,24 @@ namespace boost { namespace unordered_detail {
iterator_base erase(iterator_base r);
std::size_t erase_group(node_ptr* it, bucket_ptr bucket);
iterator_base erase_range(iterator_base r1, iterator_base r2);
// recompute_begin_bucket
//
// After an erase cached_begin_bucket_ might be left pointing to
// an empty bucket, so this is called to update it
//
// no throw
void recompute_begin_bucket(bucket_ptr b);
// This is called when a range has been erased
//
// no throw
void recompute_begin_bucket(bucket_ptr b1, bucket_ptr b2);
// no throw
float load_factor() const;
};
template <class H, class P, class A, class G, class K>

View File

@ -70,7 +70,9 @@ namespace boost { namespace unordered_detail {
inline node_ptr add_node(node_constructor& a, bucket_ptr bucket)
{
node_ptr n = a.release();
this->link_node_in_bucket(n, bucket);
node::add_to_bucket(n, *bucket);
++this->size_;
if(bucket < this->cached_begin_bucket_) this->cached_begin_bucket_ = bucket;
return n;
}
@ -328,10 +330,14 @@ namespace boost { namespace unordered_detail {
inline node_ptr add_node(node_constructor& a, bucket_ptr bucket, node_ptr pos)
{
node_ptr n = a.release();
if(BOOST_UNORDERED_BORLAND_BOOL(pos))
this->link_node(n, pos);
else
this->link_node_in_bucket(n, bucket);
if(BOOST_UNORDERED_BORLAND_BOOL(pos)) {
node::add_after_node(n, pos);
}
else {
node::add_to_bucket(n, *bucket);
if(bucket < this->cached_begin_bucket_) this->cached_begin_bucket_ = bucket;
}
++this->size_;
return n;
}

View File

@ -9,7 +9,7 @@
#include <boost/config.hpp>
#include <boost/assert.hpp>
#include <boost/unordered/detail/structure.hpp>
#include <boost/unordered/detail/node.hpp>
namespace boost { namespace unordered_detail {
@ -30,19 +30,19 @@ namespace boost { namespace unordered_detail {
template <class A, class G>
hash_table_manager<A, G>::hash_table_manager()
: structure(), allocators_() {}
: buckets_(), cached_begin_bucket_(), size_(), allocators_() {}
template <class A, class G>
hash_table_manager<A, G>::hash_table_manager(value_allocator const& a)
: structure(), allocators_(a,a) {}
: buckets_(), cached_begin_bucket_(), size_(), allocators_(a,a) {}
template <class A, class G>
hash_table_manager<A, G>::hash_table_manager(hash_table_manager const& h)
: structure(), allocators_(h.allocators_) {}
: buckets_(), cached_begin_bucket_(), size_(), allocators_(h.allocators_) {}
template <class A, class G>
hash_table_manager<A, G>::hash_table_manager(hash_table_manager& x, move_tag)
: structure(), allocators_(x.allocators_)
: buckets_(), cached_begin_bucket_(), size_(), allocators_(x.allocators_)
{
this->buckets_ = x.buckets_;
this->cached_begin_bucket_ = x.cached_begin_bucket_;
@ -56,7 +56,7 @@ namespace boost { namespace unordered_detail {
template <class A, class G>
hash_table_manager<A, G>::hash_table_manager(hash_table_manager& x, value_allocator const& a, move_tag) :
structure(), allocators_(a,a)
buckets_(), cached_begin_bucket_(), size_(), allocators_(a,a)
{
if(this->node_alloc() == x.node_alloc()) {
this->buckets_ = x.buckets_;
@ -78,8 +78,9 @@ namespace boost { namespace unordered_detail {
// no throw
template <class A, class G>
void hash_table_manager<A, G>::move(hash_table_manager& other)
inline void hash_table_manager<A, G>::move(hash_table_manager& other)
{
BOOST_ASSERT(node_alloc() == other.node_alloc());
delete_buckets();
this->buckets_ = other.buckets_;
this->cached_begin_bucket_ = other.cached_begin_bucket_;
@ -91,6 +92,131 @@ namespace boost { namespace unordered_detail {
other.bucket_count_ = 0;
}
template <class A, class G>
inline void hash_table_manager<A, G>::swap(hash_table_manager<A, G>& other)
{
BOOST_ASSERT(node_alloc() == other.node_alloc());
std::swap(buckets_, other.buckets_);
std::swap(cached_begin_bucket_, other.cached_begin_bucket_);
std::swap(size_, other.size_);
std::swap(bucket_count_, other.bucket_count_);
}
// Buckets
template <class A, class G>
inline std::size_t hash_table_manager<A, G>::bucket_count() const
{
return bucket_count_;
}
template <class A, class G>
inline std::size_t hash_table_manager<A, G>::bucket_from_hash(std::size_t hashed) const
{
return hashed % bucket_count_;
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::bucket_ptr
hash_table_manager<A, G>::bucket_ptr_from_hash(std::size_t hashed) const
{
return buckets_ + static_cast<std::ptrdiff_t>(
bucket_from_hash(hashed));
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::bucket_ptr
hash_table_manager<A, G>::buckets_begin() const
{
return buckets_;
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::bucket_ptr
hash_table_manager<A, G>::buckets_end() const
{
return buckets_ + static_cast<std::ptrdiff_t>(bucket_count_);
}
template <class A, class G>
inline std::size_t hash_table_manager<A, G>::bucket_size(std::size_t index) const
{
bucket_ptr ptr = (buckets_ + static_cast<std::ptrdiff_t>(index))->next_;
std::size_t count = 0;
while(ptr) {
++count;
ptr = next_node(ptr);
}
return count;
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::bucket_ptr
hash_table_manager<A, G>::get_bucket(std::size_t n) const
{
return buckets_ + static_cast<std::ptrdiff_t>(n);
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::node_ptr
hash_table_manager<A, G>::bucket_begin(std::size_t n) const
{
return (buckets_ + static_cast<std::ptrdiff_t>(n))->next_;
}
template <class A, class G>
inline BOOST_DEDUCED_TYPENAME hash_table_manager<A, G>::node_ptr
hash_table_manager<A, G>::bucket_end(std::size_t) const
{
return node_ptr();
}
// recompute_begin_bucket
//
// After an erase cached_begin_bucket_ might be left pointing to
// an empty bucket, so this is called to update it
//
// no throw
template <class A, class G>
inline void hash_table_manager<A, G>::recompute_begin_bucket(bucket_ptr b)
{
BOOST_ASSERT(!(b < cached_begin_bucket_));
if(b == cached_begin_bucket_)
{
if (size_ != 0) {
while (!cached_begin_bucket_->next_)
++cached_begin_bucket_;
} else {
cached_begin_bucket_ = buckets_end();
}
}
}
// This is called when a range has been erased
//
// no throw
template <class A, class G>
inline void hash_table_manager<A, G>::recompute_begin_bucket(bucket_ptr b1, bucket_ptr b2)
{
BOOST_ASSERT(!(b1 < cached_begin_bucket_) && !(b2 < b1));
BOOST_ASSERT(BOOST_UNORDERED_BORLAND_BOOL(b2->next_));
if(b1 == cached_begin_bucket_ && !b1->next_)
cached_begin_bucket_ = b2;
}
// no throw
template <class A, class G>
inline float hash_table_manager<A, G>::load_factor() const
{
BOOST_ASSERT(bucket_count_ != 0);
return static_cast<float>(size_)
/ static_cast<float>(bucket_count_);
}
// Construct/destruct
template <class A, class G>
@ -117,7 +243,7 @@ namespace boost { namespace unordered_detail {
}
template <class A, class G>
void hash_table_manager<A, G>::destruct_node(node_ptr b)
inline void hash_table_manager<A, G>::destruct_node(node_ptr b)
{
node* raw_ptr = static_cast<node*>(&*b);
BOOST_UNORDERED_DESTRUCT(&raw_ptr->value(), value_type);
@ -129,13 +255,13 @@ namespace boost { namespace unordered_detail {
// Delete and clear buckets
template <class A, class G>
void hash_table_manager<A, G>::delete_group(node_ptr first_node)
inline void hash_table_manager<A, G>::delete_group(node_ptr first_node)
{
delete_nodes(first_node, node::next_group(first_node));
}
template <class A, class G>
void hash_table_manager<A, G>::delete_nodes(node_ptr begin, node_ptr end)
inline void hash_table_manager<A, G>::delete_nodes(node_ptr begin, node_ptr end)
{
while(begin != end) {
node_ptr node = begin;
@ -145,7 +271,7 @@ namespace boost { namespace unordered_detail {
}
template <class A, class G>
void hash_table_manager<A, G>::delete_to_bucket_end(node_ptr begin)
inline void hash_table_manager<A, G>::delete_to_bucket_end(node_ptr begin)
{
while(BOOST_UNORDERED_BORLAND_BOOL(begin)) {
node_ptr node = begin;
@ -155,7 +281,7 @@ namespace boost { namespace unordered_detail {
}
template <class A, class G>
void hash_table_manager<A, G>::clear_bucket(bucket_ptr b)
inline void hash_table_manager<A, G>::clear_bucket(bucket_ptr b)
{
node_ptr node_it = b->next_;
b->next_ = node_ptr();
@ -179,7 +305,7 @@ namespace boost { namespace unordered_detail {
}
template <class A, class G>
void hash_table_manager<A, G>::delete_buckets()
inline void hash_table_manager<A, G>::delete_buckets()
{
clear();
@ -202,7 +328,8 @@ namespace boost { namespace unordered_detail {
BOOST_ASSERT(!r.is_end());
iterator_base next = r;
next.increment();
this->unlink_node(r.bucket_, r.node_);
--this->size_;
node::unlink_node(*r.bucket_, r.node_);
destruct_node(r.node_);
// r has been invalidated but its bucket is still valid
this->recompute_begin_bucket(r.bucket_, next.bucket_);
@ -210,14 +337,14 @@ namespace boost { namespace unordered_detail {
}
template <class A, class G>
std::size_t hash_table_manager<A, G>::erase_group(node_ptr* it, bucket_ptr bucket)
inline std::size_t hash_table_manager<A, G>::erase_group(node_ptr* it, bucket_ptr bucket)
{
node_ptr pos = *it;
std::size_t count = this->unlink_group(it);
std::size_t count = node::group_count(*it);
this->size_ -= count;
node::unlink_group(it);
delete_group(pos);
this->recompute_begin_bucket(bucket);
return count;
}
@ -230,7 +357,8 @@ namespace boost { namespace unordered_detail {
BOOST_ASSERT(!r1.is_end());
if (r1.bucket_ == r2.bucket_) {
this->unlink_nodes(r1.bucket_, r1.node_, r2.node_);
this->size_ -= node_count(r1.node_, r2.node_);
node::unlink_nodes(*r1.bucket_, r1.node_, r2.node_);
delete_nodes(r1.node_, r2.node_);
// No need to call recompute_begin_bucket because
@ -241,7 +369,8 @@ namespace boost { namespace unordered_detail {
else {
BOOST_ASSERT(r1.bucket_ < r2.bucket_);
this->unlink_nodes(r1.bucket_, r1.node_, node_ptr());
this->size_ -= node_count(r1.node_, node_ptr());
node::unlink_nodes(*r1.bucket_, r1.node_, node_ptr());
delete_to_bucket_end(r1.node_);
bucket_ptr i = r1.bucket_;
@ -252,7 +381,8 @@ namespace boost { namespace unordered_detail {
if(!r2.is_end()) {
node_ptr first = r2.bucket_->next_;
this->unlink_nodes(r2.bucket_, r2.node_);
this->size_ -= node_count(r2.bucket_->next_, r2.node_);
node::unlink_nodes(*r2.bucket_, r2.node_);
delete_nodes(first, r2.node_);
}
@ -265,6 +395,24 @@ namespace boost { namespace unordered_detail {
return r2;
}
////////////////////////////////////////////////////////////////////////////
// hash_iterator_base implementation
template <class BucketPtr>
inline void hash_iterator_base<BucketPtr>::increment(node_ptr node) {
while(!node) {
++bucket_;
node = bucket_->next_;
}
node_ = node;
}
template <class BucketPtr>
inline void hash_iterator_base<BucketPtr>::increment()
{
increment(next_node(node_));
}
}}
#endif

View File

@ -79,13 +79,13 @@ namespace boost { namespace unordered_detail {
}
template <class A>
void ungrouped_node_base<A>::unlink_node(bucket& b, node_ptr node)
inline void ungrouped_node_base<A>::unlink_node(bucket& b, node_ptr node)
{
unlink_nodes(b, node, next_node(node));
}
template <class A>
void ungrouped_node_base<A>::unlink_nodes(bucket& b, node_ptr begin, node_ptr end)
inline void ungrouped_node_base<A>::unlink_nodes(bucket& b, node_ptr begin, node_ptr end)
{
node_ptr* pos = &b.next_;
while(*pos != begin) pos = &next_node(*pos);
@ -93,7 +93,7 @@ namespace boost { namespace unordered_detail {
}
template <class A>
void ungrouped_node_base<A>::unlink_nodes(bucket& b, node_ptr end)
inline void ungrouped_node_base<A>::unlink_nodes(bucket& b, node_ptr end)
{
b.next_ = end;
}

View File

@ -1,220 +0,0 @@
// Copyright (C) 2003-2004 Jeremy B. Maitin-Shepard.
// Copyright (C) 2005-2009 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)
// This contains the basic data structure, apart from the actual values. There's
// no construction or deconstruction here. So this only depends on the pointer
// type.
#ifndef BOOST_UNORDERED_DETAIL_STRUCTURE_HPP_INCLUDED
#define BOOST_UNORDERED_DETAIL_STRUCTURE_HPP_INCLUDED
#include <boost/unordered/detail/node.hpp>
namespace boost { namespace unordered_detail {
////////////////////////////////////////////////////////////////////////////
// hash_structure implementation
template <class A, class G>
void hash_structure<A, G>::swap(hash_structure<A, G>& other)
{
std::swap(buckets_, other.buckets_);
std::swap(cached_begin_bucket_, other.cached_begin_bucket_);
std::swap(size_, other.size_);
std::swap(bucket_count_, other.bucket_count_);
}
template <class A, class G>
std::size_t hash_structure<A, G>::bucket_count() const
{
return bucket_count_;
}
template <class A, class G>
std::size_t hash_structure<A, G>::bucket_from_hash(std::size_t hashed) const
{
return hashed % bucket_count_;
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::bucket_ptr
hash_structure<A, G>::bucket_ptr_from_hash(std::size_t hashed) const
{
return buckets_ + static_cast<std::ptrdiff_t>(
bucket_from_hash(hashed));
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::bucket_ptr
hash_structure<A, G>::buckets_begin() const
{
return buckets_;
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::bucket_ptr
hash_structure<A, G>::buckets_end() const
{
return buckets_ + static_cast<std::ptrdiff_t>(bucket_count_);
}
template <class A, class G>
std::size_t hash_structure<A, G>::bucket_size(std::size_t index) const
{
bucket_ptr ptr = (buckets_ + static_cast<std::ptrdiff_t>(index))->next_;
std::size_t count = 0;
while(ptr) {
++count;
ptr = next_node(ptr);
}
return count;
}
// Link a node
template <class A, class G>
void hash_structure<A, G>::link_node(node_ptr n, node_ptr position)
{
node_base::add_after_node(n, position);
++size_;
}
template <class A, class G>
void hash_structure<A, G>::link_node_in_bucket(node_ptr n, bucket_ptr bucket)
{
node_base::add_to_bucket(n, *bucket);
++size_;
if(bucket < cached_begin_bucket_) cached_begin_bucket_ = bucket;
}
template <class A, class G>
void hash_structure<A, G>::unlink_node(bucket_ptr bucket, node_ptr pos)
{
--size_;
node_base::unlink_node(*bucket, pos);
}
template <class A, class G>
void hash_structure<A, G>::unlink_nodes(
bucket_ptr bucket, node_ptr begin, node_ptr end)
{
size_ -= node_count(begin, end);
node_base::unlink_nodes(*bucket, begin, end);
}
template <class A, class G>
void hash_structure<A, G>::unlink_nodes(bucket_ptr bucket, node_ptr end)
{
size_ -= node_count(bucket->next_, end);
node_base::unlink_nodes(*bucket, end);
}
template <class A, class G>
std::size_t hash_structure<A, G>::unlink_group(node_ptr* pos)
{
std::size_t count = node_base::group_count(*pos);
size_ -= count;
node_base::unlink_group(pos);
return count;
}
template <class A, class G>
void hash_structure<A, G>::link_group(
node_ptr n, bucket_ptr bucket, std::size_t count)
{
node_base::add_group_to_bucket(n, *bucket);
size_ += count;
if(bucket < cached_begin_bucket_) cached_begin_bucket_ = bucket;
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::bucket_ptr
hash_structure<A, G>::get_bucket(std::size_t n) const
{
return buckets_ + static_cast<std::ptrdiff_t>(n);
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::node_ptr
hash_structure<A, G>::bucket_begin(std::size_t n) const
{
return (buckets_ + static_cast<std::ptrdiff_t>(n))->next_;
}
template <class A, class G>
BOOST_DEDUCED_TYPENAME hash_structure<A, G>::node_ptr
hash_structure<A, G>::bucket_end(std::size_t) const
{
return node_ptr();
}
// recompute_begin_bucket
//
// After an erase cached_begin_bucket_ might be left pointing to
// an empty bucket, so this is called to update it
//
// no throw
template <class A, class G>
void hash_structure<A, G>::recompute_begin_bucket(bucket_ptr b)
{
BOOST_ASSERT(!(b < cached_begin_bucket_));
if(b == cached_begin_bucket_)
{
if (size_ != 0) {
while (!cached_begin_bucket_->next_)
++cached_begin_bucket_;
} else {
cached_begin_bucket_ = buckets_end();
}
}
}
// This is called when a range has been erased
//
// no throw
template <class A, class G>
void hash_structure<A, G>::recompute_begin_bucket(bucket_ptr b1, bucket_ptr b2)
{
BOOST_ASSERT(!(b1 < cached_begin_bucket_) && !(b2 < b1));
BOOST_ASSERT(BOOST_UNORDERED_BORLAND_BOOL(b2->next_));
if(b1 == cached_begin_bucket_ && !b1->next_)
cached_begin_bucket_ = b2;
}
// no throw
template <class A, class G>
inline float hash_structure<A, G>::load_factor() const
{
BOOST_ASSERT(bucket_count_ != 0);
return static_cast<float>(size_)
/ static_cast<float>(bucket_count_);
}
////////////////////////////////////////////////////////////////////////////
// hash_iterator_base implementation
template <class BucketPtr>
inline void hash_iterator_base<BucketPtr>::increment(node_ptr node) {
while(!node) {
++bucket_;
node = bucket_->next_;
}
node_ = node;
}
template <class BucketPtr>
inline void hash_iterator_base<BucketPtr>::increment()
{
increment(next_node(node_));
}
}}
#endif

View File

@ -327,7 +327,7 @@ namespace boost { namespace unordered_detail {
// basic exception safety
template <class H, class P, class A, class G, class K>
bool hash_table<H, P, A, G, K>
inline bool hash_table<H, P, A, G, K>
::reserve(std::size_t n)
{
bool need_to_reserve = n >= this->max_load_;
@ -339,7 +339,7 @@ namespace boost { namespace unordered_detail {
// basic exception safety
template <class H, class P, class A, class G, class K>
bool hash_table<H, P, A, G, K>
inline bool hash_table<H, P, A, G, K>
::reserve_for_insert(std::size_t n)
{
bool need_to_reserve = n >= this->max_load_;
@ -419,8 +419,12 @@ namespace boost { namespace unordered_detail {
hf(extractor::extract(node::get_value(src_bucket->next_))));
node_ptr n = src_bucket->next_;
std::size_t count = this->unlink_group(&src_bucket->next_);
dst.link_group(n, dst_bucket, count);
std::size_t count = node::group_count(src_bucket->next_);
this->size_ -= count;
dst.size_ += count;
node::unlink_group(&src_bucket->next_);
node::add_group_to_bucket(n, *dst_bucket);
if(dst_bucket < dst.cached_begin_bucket_) dst.cached_begin_bucket_ = dst_bucket;
}
}
}
@ -454,11 +458,14 @@ namespace boost { namespace unordered_detail {
a.construct(node::get_value(it));
node_ptr n = a.release();
dst.link_node_in_bucket(n, dst_bucket);
node::add_to_bucket(n, *dst_bucket);
++dst.size_;
if(dst_bucket < dst.cached_begin_bucket_) dst.cached_begin_bucket_ = dst_bucket;
for(it = next_node(it); it != group_end; it = next_node(it)) {
a.construct(node::get_value(it));
dst.link_node(a.release(), n);
node::add_after_node(a.release(), n);
++dst.size_;
}
}
}