1 /* 2 * Copyright (c) 1996,1997 3 * Silicon Graphics Computer Systems, Inc. 4 * 5 * Permission to use, copy, modify, distribute and sell this software 6 * and its documentation for any purpose is hereby granted without fee, 7 * provided that the above copyright notice appear in all copies and 8 * that both that copyright notice and this permission notice appear 9 * in supporting documentation. Silicon Graphics makes no 10 * representations about the suitability of this software for any 11 * purpose. It is provided "as is" without express or implied warranty. 12 * 13 * 14 * Copyright (c) 1994 15 * Hewlett-Packard Company 16 * 17 * Permission to use, copy, modify, distribute and sell this software 18 * and its documentation for any purpose is hereby granted without fee, 19 * provided that the above copyright notice appear in all copies and 20 * that both that copyright notice and this permission notice appear 21 * in supporting documentation. Hewlett-Packard Company makes no 22 * representations about the suitability of this software for any 23 * purpose. It is provided "as is" without express or implied warranty. 24 * 25 */ 26 27 /* NOTE: This is an internal header file, included by other STL headers. 28 * You should not attempt to use it directly. 29 */ 30 31 #ifndef __SGI_STL_INTERNAL_HASHTABLE_H 32 #define __SGI_STL_INTERNAL_HASHTABLE_H 33 34 // Hashtable class, used to implement the hashed associative containers 35 // hash_set, hash_map, hash_multiset, and hash_multimap. 36 37 #include <stl_algobase.h> 38 #include <stl_alloc.h> 39 #include <stl_construct.h> 40 #include <stl_tempbuf.h> 41 #include <stl_algo.h> 42 #include <stl_uninitialized.h> 43 #include <stl_function.h> 44 #include <stl_vector.h> 45 #include <stl_hash_fun.h> 46 47 __STL_BEGIN_NAMESPACE 48 49 template <class _Val> 50 struct _Hashtable_node 51 { 52 _Hashtable_node* _M_next; 53 _Val _M_val; 54 }; 55 56 template <class _Val, class _Key, class _HashFcn, 57 class _ExtractKey, class _EqualKey, class _Alloc = alloc> 58 class hashtable; 59 60 template <class _Val, class _Key, class _HashFcn, 61 class _ExtractKey, class _EqualKey, class _Alloc> 62 struct _Hashtable_iterator; 63 64 template <class _Val, class _Key, class _HashFcn, 65 class _ExtractKey, class _EqualKey, class _Alloc> 66 struct _Hashtable_const_iterator; 67 68 template <class _Val, class _Key, class _HashFcn, 69 class _ExtractKey, class _EqualKey, class _Alloc> 70 struct _Hashtable_iterator { 71 typedef hashtable<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey,_Alloc> 72 _Hashtable; 73 typedef _Hashtable_iterator<_Val, _Key, _HashFcn, 74 _ExtractKey, _EqualKey, _Alloc> 75 iterator; 76 typedef _Hashtable_const_iterator<_Val, _Key, _HashFcn, 77 _ExtractKey, _EqualKey, _Alloc> 78 const_iterator; 79 typedef _Hashtable_node<_Val> _Node; 80 81 typedef forward_iterator_tag iterator_category; 82 typedef _Val value_type; 83 typedef ptrdiff_t difference_type; 84 typedef size_t size_type; 85 typedef _Val& reference; 86 typedef _Val* pointer; 87 88 _Node* _M_cur; 89 _Hashtable* _M_ht; 90 91 _Hashtable_iterator(_Node* __n, _Hashtable* __tab) 92 : _M_cur(__n), _M_ht(__tab) {} 93 _Hashtable_iterator() {} 94 reference operator*() const { return _M_cur->_M_val; } 95 #ifndef __SGI_STL_NO_ARROW_OPERATOR 96 pointer operator->() const { return &(operator*()); } 97 #endif /* __SGI_STL_NO_ARROW_OPERATOR */ 98 iterator& operator++(); 99 iterator operator++(int); 100 bool operator==(const iterator& __it) const 101 { return _M_cur == __it._M_cur; } 102 bool operator!=(const iterator& __it) const 103 { return _M_cur != __it._M_cur; } 104 }; 105 106 107 template <class _Val, class _Key, class _HashFcn, 108 class _ExtractKey, class _EqualKey, class _Alloc> 109 struct _Hashtable_const_iterator { 110 typedef hashtable<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey,_Alloc> 111 _Hashtable; 112 typedef _Hashtable_iterator<_Val,_Key,_HashFcn, 113 _ExtractKey,_EqualKey,_Alloc> 114 iterator; 115 typedef _Hashtable_const_iterator<_Val, _Key, _HashFcn, 116 _ExtractKey, _EqualKey, _Alloc> 117 const_iterator; 118 typedef _Hashtable_node<_Val> _Node; 119 120 typedef forward_iterator_tag iterator_category; 121 typedef _Val value_type; 122 typedef ptrdiff_t difference_type; 123 typedef size_t size_type; 124 typedef const _Val& reference; 125 typedef const _Val* pointer; 126 127 const _Node* _M_cur; 128 const _Hashtable* _M_ht; 129 130 _Hashtable_const_iterator(const _Node* __n, const _Hashtable* __tab) 131 : _M_cur(__n), _M_ht(__tab) {} 132 _Hashtable_const_iterator() {} 133 _Hashtable_const_iterator(const iterator& __it) 134 : _M_cur(__it._M_cur), _M_ht(__it._M_ht) {} 135 reference operator*() const { return _M_cur->_M_val; } 136 #ifndef __SGI_STL_NO_ARROW_OPERATOR 137 pointer operator->() const { return &(operator*()); } 138 #endif /* __SGI_STL_NO_ARROW_OPERATOR */ 139 const_iterator& operator++(); 140 const_iterator operator++(int); 141 bool operator==(const const_iterator& __it) const 142 { return _M_cur == __it._M_cur; } 143 bool operator!=(const const_iterator& __it) const 144 { return _M_cur != __it._M_cur; } 145 }; 146 147 // Note: assumes long is at least 32 bits. 148 static const int __stl_num_primes = 28; 149 static const unsigned long __stl_prime_list[__stl_num_primes] = 150 { 151 53ul, 97ul, 193ul, 389ul, 769ul, 152 1543ul, 3079ul, 6151ul, 12289ul, 24593ul, 153 49157ul, 98317ul, 196613ul, 393241ul, 786433ul, 154 1572869ul, 3145739ul, 6291469ul, 12582917ul, 25165843ul, 155 50331653ul, 100663319ul, 201326611ul, 402653189ul, 805306457ul, 156 1610612741ul, 3221225473ul, 4294967291ul 157 }; 158 159 inline unsigned long __stl_next_prime(unsigned long __n) 160 { 161 const unsigned long* __first = __stl_prime_list; 162 const unsigned long* __last = __stl_prime_list + __stl_num_primes; 163 const unsigned long* pos = lower_bound(__first, __last, __n); 164 return pos == __last ? *(__last - 1) : *pos; 165 } 166 167 // Forward declaration of operator==. 168 169 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 170 class hashtable; 171 172 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 173 bool operator==(const hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>& __ht1, 174 const hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>& __ht2); 175 176 177 // Hashtables handle allocators a bit differently than other containers 178 // do. If we're using standard-conforming allocators, then a hashtable 179 // unconditionally has a member variable to hold its allocator, even if 180 // it so happens that all instances of the allocator type are identical. 181 // This is because, for hashtables, this extra storage is negligible. 182 // Additionally, a base class wouldn't serve any other purposes; it 183 // wouldn't, for example, simplify the exception-handling code. 184 185 template <class _Val, class _Key, class _HashFcn, 186 class _ExtractKey, class _EqualKey, class _Alloc> 187 class hashtable { 188 public: 189 typedef _Key key_type; 190 typedef _Val value_type; 191 typedef _HashFcn hasher; 192 typedef _EqualKey key_equal; 193 194 typedef size_t size_type; 195 typedef ptrdiff_t difference_type; 196 typedef value_type* pointer; 197 typedef const value_type* const_pointer; 198 typedef value_type& reference; 199 typedef const value_type& const_reference; 200 201 hasher hash_funct() const { return _M_hash; } 202 key_equal key_eq() const { return _M_equals; } 203 204 private: 205 typedef _Hashtable_node<_Val> _Node; 206 207 #ifdef __STL_USE_STD_ALLOCATORS 208 public: 209 typedef typename _Alloc_traits<_Val,_Alloc>::allocator_type allocator_type; 210 allocator_type get_allocator() const { return _M_node_allocator; } 211 private: 212 typename _Alloc_traits<_Node, _Alloc>::allocator_type _M_node_allocator; 213 _Node* _M_get_node() { return _M_node_allocator.allocate(1); } 214 void _M_put_node(_Node* __p) { _M_node_allocator.deallocate(__p, 1); } 215 # define __HASH_ALLOC_INIT(__a) _M_node_allocator(__a), 216 #else /* __STL_USE_STD_ALLOCATORS */ 217 public: 218 typedef _Alloc allocator_type; 219 allocator_type get_allocator() const { return allocator_type(); } 220 private: 221 typedef simple_alloc<_Node, _Alloc> _M_node_allocator_type; 222 _Node* _M_get_node() { return _M_node_allocator_type::allocate(1); } 223 void _M_put_node(_Node* __p) { _M_node_allocator_type::deallocate(__p, 1); } 224 # define __HASH_ALLOC_INIT(__a) 225 #endif /* __STL_USE_STD_ALLOCATORS */ 226 227 private: 228 hasher _M_hash; 229 key_equal _M_equals; 230 _ExtractKey _M_get_key; 231 vector<_Node*,_Alloc> _M_buckets; 232 size_type _M_num_elements; 233 234 public: 235 typedef _Hashtable_iterator<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey,_Alloc> 236 iterator; 237 typedef _Hashtable_const_iterator<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey, 238 _Alloc> 239 const_iterator; 240 241 friend struct 242 _Hashtable_iterator<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey,_Alloc>; 243 friend struct 244 _Hashtable_const_iterator<_Val,_Key,_HashFcn,_ExtractKey,_EqualKey,_Alloc>; 245 246 public: 247 hashtable(size_type __n, 248 const _HashFcn& __hf, 249 const _EqualKey& __eql, 250 const _ExtractKey& __ext, 251 const allocator_type& __a = allocator_type()) 252 : __HASH_ALLOC_INIT(__a) 253 _M_hash(__hf), 254 _M_equals(__eql), 255 _M_get_key(__ext), 256 _M_buckets(__a), 257 _M_num_elements(0) 258 { 259 _M_initialize_buckets(__n); 260 } 261 262 hashtable(size_type __n, 263 const _HashFcn& __hf, 264 const _EqualKey& __eql, 265 const allocator_type& __a = allocator_type()) 266 : __HASH_ALLOC_INIT(__a) 267 _M_hash(__hf), 268 _M_equals(__eql), 269 _M_get_key(_ExtractKey()), 270 _M_buckets(__a), 271 _M_num_elements(0) 272 { 273 _M_initialize_buckets(__n); 274 } 275 276 hashtable(const hashtable& __ht) 277 : __HASH_ALLOC_INIT(__ht.get_allocator()) 278 _M_hash(__ht._M_hash), 279 _M_equals(__ht._M_equals), 280 _M_get_key(__ht._M_get_key), 281 _M_buckets(__ht.get_allocator()), 282 _M_num_elements(0) 283 { 284 _M_copy_from(__ht); 285 } 286 287 #undef __HASH_ALLOC_INIT 288 289 hashtable& operator= (const hashtable& __ht) 290 { 291 if (&__ht != this) { 292 clear(); 293 _M_hash = __ht._M_hash; 294 _M_equals = __ht._M_equals; 295 _M_get_key = __ht._M_get_key; 296 _M_copy_from(__ht); 297 } 298 return *this; 299 } 300 301 ~hashtable() { clear(); } 302 303 size_type size() const { return _M_num_elements; } 304 size_type max_size() const { return size_type(-1); } 305 bool empty() const { return size() == 0; } 306 307 void swap(hashtable& __ht) 308 { 309 __STD::swap(_M_hash, __ht._M_hash); 310 __STD::swap(_M_equals, __ht._M_equals); 311 __STD::swap(_M_get_key, __ht._M_get_key); 312 _M_buckets.swap(__ht._M_buckets); 313 __STD::swap(_M_num_elements, __ht._M_num_elements); 314 } 315 316 iterator begin() 317 { 318 for (size_type __n = 0; __n < _M_buckets.size(); ++__n) 319 if (_M_buckets[__n]) 320 return iterator(_M_buckets[__n], this); 321 return end(); 322 } 323 324 iterator end() { return iterator(0, this); } 325 326 const_iterator begin() const 327 { 328 for (size_type __n = 0; __n < _M_buckets.size(); ++__n) 329 if (_M_buckets[__n]) 330 return const_iterator(_M_buckets[__n], this); 331 return end(); 332 } 333 334 const_iterator end() const { return const_iterator(0, this); } 335 336 friend bool 337 operator== __STL_NULL_TMPL_ARGS (const hashtable&, const hashtable&); 338 339 public: 340 341 size_type bucket_count() const { return _M_buckets.size(); } 342 343 size_type max_bucket_count() const 344 { return __stl_prime_list[__stl_num_primes - 1]; } 345 346 size_type elems_in_bucket(size_type __bucket) const 347 { 348 size_type __result = 0; 349 for (_Node* __cur = _M_buckets[__bucket]; __cur; __cur = __cur->_M_next) 350 __result += 1; 351 return __result; 352 } 353 354 pair<iterator, bool> insert_unique(const value_type& __obj) 355 { 356 resize(_M_num_elements + 1); 357 return insert_unique_noresize(__obj); 358 } 359 360 iterator insert_equal(const value_type& __obj) 361 { 362 resize(_M_num_elements + 1); 363 return insert_equal_noresize(__obj); 364 } 365 366 pair<iterator, bool> insert_unique_noresize(const value_type& __obj); 367 iterator insert_equal_noresize(const value_type& __obj); 368 369 #ifdef __STL_MEMBER_TEMPLATES 370 template <class _InputIterator> 371 void insert_unique(_InputIterator __f, _InputIterator __l) 372 { 373 insert_unique(__f, __l, __ITERATOR_CATEGORY(__f)); 374 } 375 376 template <class _InputIterator> 377 void insert_equal(_InputIterator __f, _InputIterator __l) 378 { 379 insert_equal(__f, __l, __ITERATOR_CATEGORY(__f)); 380 } 381 382 template <class _InputIterator> 383 void insert_unique(_InputIterator __f, _InputIterator __l, 384 input_iterator_tag) 385 { 386 for ( ; __f != __l; ++__f) 387 insert_unique(*__f); 388 } 389 390 template <class _InputIterator> 391 void insert_equal(_InputIterator __f, _InputIterator __l, 392 input_iterator_tag) 393 { 394 for ( ; __f != __l; ++__f) 395 insert_equal(*__f); 396 } 397 398 template <class _ForwardIterator> 399 void insert_unique(_ForwardIterator __f, _ForwardIterator __l, 400 forward_iterator_tag) 401 { 402 size_type __n = 0; 403 distance(__f, __l, __n); 404 resize(_M_num_elements + __n); 405 for ( ; __n > 0; --__n, ++__f) 406 insert_unique_noresize(*__f); 407 } 408 409 template <class _ForwardIterator> 410 void insert_equal(_ForwardIterator __f, _ForwardIterator __l, 411 forward_iterator_tag) 412 { 413 size_type __n = 0; 414 distance(__f, __l, __n); 415 resize(_M_num_elements + __n); 416 for ( ; __n > 0; --__n, ++__f) 417 insert_equal_noresize(*__f); 418 } 419 420 #else /* __STL_MEMBER_TEMPLATES */ 421 void insert_unique(const value_type* __f, const value_type* __l) 422 { 423 size_type __n = __l - __f; 424 resize(_M_num_elements + __n); 425 for ( ; __n > 0; --__n, ++__f) 426 insert_unique_noresize(*__f); 427 } 428 429 void insert_equal(const value_type* __f, const value_type* __l) 430 { 431 size_type __n = __l - __f; 432 resize(_M_num_elements + __n); 433 for ( ; __n > 0; --__n, ++__f) 434 insert_equal_noresize(*__f); 435 } 436 437 void insert_unique(const_iterator __f, const_iterator __l) 438 { 439 size_type __n = 0; 440 distance(__f, __l, __n); 441 resize(_M_num_elements + __n); 442 for ( ; __n > 0; --__n, ++__f) 443 insert_unique_noresize(*__f); 444 } 445 446 void insert_equal(const_iterator __f, const_iterator __l) 447 { 448 size_type __n = 0; 449 distance(__f, __l, __n); 450 resize(_M_num_elements + __n); 451 for ( ; __n > 0; --__n, ++__f) 452 insert_equal_noresize(*__f); 453 } 454 #endif /*__STL_MEMBER_TEMPLATES */ 455 456 reference find_or_insert(const value_type& __obj); 457 458 iterator find(const key_type& __key) 459 { 460 size_type __n = _M_bkt_num_key(__key); 461 _Node* __first; 462 for ( __first = _M_buckets[__n]; 463 __first && !_M_equals(_M_get_key(__first->_M_val), __key); 464 __first = __first->_M_next) 465 {} 466 return iterator(__first, this); 467 } 468 469 const_iterator find(const key_type& __key) const 470 { 471 size_type __n = _M_bkt_num_key(__key); 472 const _Node* __first; 473 for ( __first = _M_buckets[__n]; 474 __first && !_M_equals(_M_get_key(__first->_M_val), __key); 475 __first = __first->_M_next) 476 {} 477 return const_iterator(__first, this); 478 } 479 480 size_type count(const key_type& __key) const 481 { 482 const size_type __n = _M_bkt_num_key(__key); 483 size_type __result = 0; 484 485 for (const _Node* __cur = _M_buckets[__n]; __cur; __cur = __cur->_M_next) 486 if (_M_equals(_M_get_key(__cur->_M_val), __key)) 487 ++__result; 488 return __result; 489 } 490 491 pair<iterator, iterator> 492 equal_range(const key_type& __key); 493 494 pair<const_iterator, const_iterator> 495 equal_range(const key_type& __key) const; 496 497 size_type erase(const key_type& __key); 498 void erase(const iterator& __it); 499 void erase(iterator __first, iterator __last); 500 501 void erase(const const_iterator& __it); 502 void erase(const_iterator __first, const_iterator __last); 503 504 void resize(size_type __num_elements_hint); 505 void clear(); 506 507 private: 508 size_type _M_next_size(size_type __n) const 509 { return __stl_next_prime(__n); } 510 511 void _M_initialize_buckets(size_type __n) 512 { 513 const size_type __n_buckets = _M_next_size(__n); 514 _M_buckets.reserve(__n_buckets); 515 _M_buckets.insert(_M_buckets.end(), __n_buckets, (_Node*) 0); 516 _M_num_elements = 0; 517 } 518 519 size_type _M_bkt_num_key(const key_type& __key) const 520 { 521 return _M_bkt_num_key(__key, _M_buckets.size()); 522 } 523 524 size_type _M_bkt_num(const value_type& __obj) const 525 { 526 return _M_bkt_num_key(_M_get_key(__obj)); 527 } 528 529 size_type _M_bkt_num_key(const key_type& __key, size_t __n) const 530 { 531 return _M_hash(__key) % __n; 532 } 533 534 size_type _M_bkt_num(const value_type& __obj, size_t __n) const 535 { 536 return _M_bkt_num_key(_M_get_key(__obj), __n); 537 } 538 539 _Node* _M_new_node(const value_type& __obj) 540 { 541 _Node* __n = _M_get_node(); 542 __n->_M_next = 0; 543 __STL_TRY { 544 construct(&__n->_M_val, __obj); 545 return __n; 546 } 547 __STL_UNWIND(_M_put_node(__n)); 548 } 549 550 void _M_delete_node(_Node* __n) 551 { 552 destroy(&__n->_M_val); 553 _M_put_node(__n); 554 } 555 556 void _M_erase_bucket(const size_type __n, _Node* __first, _Node* __last); 557 void _M_erase_bucket(const size_type __n, _Node* __last); 558 559 void _M_copy_from(const hashtable& __ht); 560 561 }; 562 563 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 564 class _All> 565 _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>& 566 _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>::operator++() 567 { 568 const _Node* __old = _M_cur; 569 _M_cur = _M_cur->_M_next; 570 if (!_M_cur) { 571 size_type __bucket = _M_ht->_M_bkt_num(__old->_M_val); 572 while (!_M_cur && ++__bucket < _M_ht->_M_buckets.size()) 573 _M_cur = _M_ht->_M_buckets[__bucket]; 574 } 575 return *this; 576 } 577 578 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 579 class _All> 580 inline _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All> 581 _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>::operator++(int) 582 { 583 iterator __tmp = *this; 584 ++*this; 585 return __tmp; 586 } 587 588 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 589 class _All> 590 _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>& 591 _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>::operator++() 592 { 593 const _Node* __old = _M_cur; 594 _M_cur = _M_cur->_M_next; 595 if (!_M_cur) { 596 size_type __bucket = _M_ht->_M_bkt_num(__old->_M_val); 597 while (!_M_cur && ++__bucket < _M_ht->_M_buckets.size()) 598 _M_cur = _M_ht->_M_buckets[__bucket]; 599 } 600 return *this; 601 } 602 603 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 604 class _All> 605 inline _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All> 606 _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>::operator++(int) 607 { 608 const_iterator __tmp = *this; 609 ++*this; 610 return __tmp; 611 } 612 613 #ifndef __STL_CLASS_PARTIAL_SPECIALIZATION 614 615 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 616 class _All> 617 inline forward_iterator_tag 618 iterator_category(const _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>&) 619 { 620 return forward_iterator_tag(); 621 } 622 623 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 624 class _All> 625 inline _Val* 626 value_type(const _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>&) 627 { 628 return (_Val*) 0; 629 } 630 631 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 632 class _All> 633 inline hashtable<_Val,_Key,_HF,_ExK,_EqK,_All>::difference_type* 634 distance_type(const _Hashtable_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>&) 635 { 636 return (hashtable<_Val,_Key,_HF,_ExK,_EqK,_All>::difference_type*) 0; 637 } 638 639 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 640 class _All> 641 inline forward_iterator_tag 642 iterator_category(const _Hashtable_const_iterator<_Val,_Key,_HF, 643 _ExK,_EqK,_All>&) 644 { 645 return forward_iterator_tag(); 646 } 647 648 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 649 class _All> 650 inline _Val* 651 value_type(const _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>&) 652 { 653 return (_Val*) 0; 654 } 655 656 template <class _Val, class _Key, class _HF, class _ExK, class _EqK, 657 class _All> 658 inline hashtable<_Val,_Key,_HF,_ExK,_EqK,_All>::difference_type* 659 distance_type(const _Hashtable_const_iterator<_Val,_Key,_HF,_ExK,_EqK,_All>&) 660 { 661 return (hashtable<_Val,_Key,_HF,_ExK,_EqK,_All>::difference_type*) 0; 662 } 663 664 #endif /* __STL_CLASS_PARTIAL_SPECIALIZATION */ 665 666 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 667 inline bool operator==(const hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>& __ht1, 668 const hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>& __ht2) 669 { 670 typedef typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::_Node _Node; 671 if (__ht1._M_buckets.size() != __ht2._M_buckets.size()) 672 return false; 673 for (int __n = 0; __n < __ht1._M_buckets.size(); ++__n) { 674 _Node* __cur1 = __ht1._M_buckets[__n]; 675 _Node* __cur2 = __ht2._M_buckets[__n]; 676 for ( ; __cur1 && __cur2 && __cur1->_M_val == __cur2->_M_val; 677 __cur1 = __cur1->_M_next, __cur2 = __cur2->_M_next) 678 {} 679 if (__cur1 || __cur2) 680 return false; 681 } 682 return true; 683 } 684 685 #ifdef __STL_FUNCTION_TMPL_PARTIAL_ORDER 686 687 template <class _Val, class _Key, class _HF, class _Extract, class _EqKey, 688 class _All> 689 inline void swap(hashtable<_Val, _Key, _HF, _Extract, _EqKey, _All>& __ht1, 690 hashtable<_Val, _Key, _HF, _Extract, _EqKey, _All>& __ht2) { 691 __ht1.swap(__ht2); 692 } 693 694 #endif /* __STL_FUNCTION_TMPL_PARTIAL_ORDER */ 695 696 697 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 698 pair<typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::iterator, bool> 699 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 700 ::insert_unique_noresize(const value_type& __obj) 701 { 702 const size_type __n = _M_bkt_num(__obj); 703 _Node* __first = _M_buckets[__n]; 704 705 for (_Node* __cur = __first; __cur; __cur = __cur->_M_next) 706 if (_M_equals(_M_get_key(__cur->_M_val), _M_get_key(__obj))) 707 return pair<iterator, bool>(iterator(__cur, this), false); 708 709 _Node* __tmp = _M_new_node(__obj); 710 __tmp->_M_next = __first; 711 _M_buckets[__n] = __tmp; 712 ++_M_num_elements; 713 return pair<iterator, bool>(iterator(__tmp, this), true); 714 } 715 716 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 717 typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::iterator 718 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 719 ::insert_equal_noresize(const value_type& __obj) 720 { 721 const size_type __n = _M_bkt_num(__obj); 722 _Node* __first = _M_buckets[__n]; 723 724 for (_Node* __cur = __first; __cur; __cur = __cur->_M_next) 725 if (_M_equals(_M_get_key(__cur->_M_val), _M_get_key(__obj))) { 726 _Node* __tmp = _M_new_node(__obj); 727 __tmp->_M_next = __cur->_M_next; 728 __cur->_M_next = __tmp; 729 ++_M_num_elements; 730 return iterator(__tmp, this); 731 } 732 733 _Node* __tmp = _M_new_node(__obj); 734 __tmp->_M_next = __first; 735 _M_buckets[__n] = __tmp; 736 ++_M_num_elements; 737 return iterator(__tmp, this); 738 } 739 740 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 741 typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::reference 742 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::find_or_insert(const value_type& __obj) 743 { 744 resize(_M_num_elements + 1); 745 746 size_type __n = _M_bkt_num(__obj); 747 _Node* __first = _M_buckets[__n]; 748 749 for (_Node* __cur = __first; __cur; __cur = __cur->_M_next) 750 if (_M_equals(_M_get_key(__cur->_M_val), _M_get_key(__obj))) 751 return __cur->_M_val; 752 753 _Node* __tmp = _M_new_node(__obj); 754 __tmp->_M_next = __first; 755 _M_buckets[__n] = __tmp; 756 ++_M_num_elements; 757 return __tmp->_M_val; 758 } 759 760 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 761 pair<typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::iterator, 762 typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::iterator> 763 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::equal_range(const key_type& __key) 764 { 765 typedef pair<iterator, iterator> _Pii; 766 const size_type __n = _M_bkt_num_key(__key); 767 768 for (_Node* __first = _M_buckets[__n]; __first; __first = __first->_M_next) 769 if (_M_equals(_M_get_key(__first->_M_val), __key)) { 770 for (_Node* __cur = __first->_M_next; __cur; __cur = __cur->_M_next) 771 if (!_M_equals(_M_get_key(__cur->_M_val), __key)) 772 return _Pii(iterator(__first, this), iterator(__cur, this)); 773 for (size_type __m = __n + 1; __m < _M_buckets.size(); ++__m) 774 if (_M_buckets[__m]) 775 return _Pii(iterator(__first, this), 776 iterator(_M_buckets[__m], this)); 777 return _Pii(iterator(__first, this), end()); 778 } 779 return _Pii(end(), end()); 780 } 781 782 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 783 pair<typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::const_iterator, 784 typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::const_iterator> 785 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 786 ::equal_range(const key_type& __key) const 787 { 788 typedef pair<const_iterator, const_iterator> _Pii; 789 const size_type __n = _M_bkt_num_key(__key); 790 791 for (const _Node* __first = _M_buckets[__n] ; 792 __first; 793 __first = __first->_M_next) { 794 if (_M_equals(_M_get_key(__first->_M_val), __key)) { 795 for (const _Node* __cur = __first->_M_next; 796 __cur; 797 __cur = __cur->_M_next) 798 if (!_M_equals(_M_get_key(__cur->_M_val), __key)) 799 return _Pii(const_iterator(__first, this), 800 const_iterator(__cur, this)); 801 for (size_type __m = __n + 1; __m < _M_buckets.size(); ++__m) 802 if (_M_buckets[__m]) 803 return _Pii(const_iterator(__first, this), 804 const_iterator(_M_buckets[__m], this)); 805 return _Pii(const_iterator(__first, this), end()); 806 } 807 } 808 return _Pii(end(), end()); 809 } 810 811 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 812 typename hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::size_type 813 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::erase(const key_type& __key) 814 { 815 const size_type __n = _M_bkt_num_key(__key); 816 _Node* __first = _M_buckets[__n]; 817 size_type __erased = 0; 818 819 if (__first) { 820 _Node* __cur = __first; 821 _Node* __next = __cur->_M_next; 822 while (__next) { 823 if (_M_equals(_M_get_key(__next->_M_val), __key)) { 824 __cur->_M_next = __next->_M_next; 825 _M_delete_node(__next); 826 __next = __cur->_M_next; 827 ++__erased; 828 --_M_num_elements; 829 } 830 else { 831 __cur = __next; 832 __next = __cur->_M_next; 833 } 834 } 835 if (_M_equals(_M_get_key(__first->_M_val), __key)) { 836 _M_buckets[__n] = __first->_M_next; 837 _M_delete_node(__first); 838 ++__erased; 839 --_M_num_elements; 840 } 841 } 842 return __erased; 843 } 844 845 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 846 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::erase(const iterator& __it) 847 { 848 if (_Node* const __p = __it._M_cur) { 849 const size_type __n = _M_bkt_num(__p->_M_val); 850 _Node* __cur = _M_buckets[__n]; 851 852 if (__cur == __p) { 853 _M_buckets[__n] = __cur->_M_next; 854 _M_delete_node(__cur); 855 --_M_num_elements; 856 } 857 else { 858 _Node* __next = __cur->_M_next; 859 while (__next) { 860 if (__next == __p) { 861 __cur->_M_next = __next->_M_next; 862 _M_delete_node(__next); 863 --_M_num_elements; 864 break; 865 } 866 else { 867 __cur = __next; 868 __next = __cur->_M_next; 869 } 870 } 871 } 872 } 873 } 874 875 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 876 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 877 ::erase(iterator __first, iterator __last) 878 { 879 size_type __f_bucket = __first._M_cur ? 880 _M_bkt_num(__first._M_cur->_M_val) : _M_buckets.size(); 881 size_type __l_bucket = __last._M_cur ? 882 _M_bkt_num(__last._M_cur->_M_val) : _M_buckets.size(); 883 884 if (__first._M_cur == __last._M_cur) 885 return; 886 else if (__f_bucket == __l_bucket) 887 _M_erase_bucket(__f_bucket, __first._M_cur, __last._M_cur); 888 else { 889 _M_erase_bucket(__f_bucket, __first._M_cur, 0); 890 for (size_type __n = __f_bucket + 1; __n < __l_bucket; ++__n) 891 _M_erase_bucket(__n, 0); 892 if (__l_bucket != _M_buckets.size()) 893 _M_erase_bucket(__l_bucket, __last._M_cur); 894 } 895 } 896 897 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 898 inline void 899 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::erase(const_iterator __first, 900 const_iterator __last) 901 { 902 erase(iterator(const_cast<_Node*>(__first._M_cur), 903 const_cast<hashtable*>(__first._M_ht)), 904 iterator(const_cast<_Node*>(__last._M_cur), 905 const_cast<hashtable*>(__last._M_ht))); 906 } 907 908 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 909 inline void 910 hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::erase(const const_iterator& __it) 911 { 912 erase(iterator(const_cast<_Node*>(__it._M_cur), 913 const_cast<hashtable*>(__it._M_ht))); 914 } 915 916 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 917 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 918 ::resize(size_type __num_elements_hint) 919 { 920 const size_type __old_n = _M_buckets.size(); 921 if (__num_elements_hint > __old_n) { 922 const size_type __n = _M_next_size(__num_elements_hint); 923 if (__n > __old_n) { 924 vector<_Node*, _All> __tmp(__n, (_Node*)(0), 925 _M_buckets.get_allocator()); 926 __STL_TRY { 927 for (size_type __bucket = 0; __bucket < __old_n; ++__bucket) { 928 _Node* __first = _M_buckets[__bucket]; 929 while (__first) { 930 size_type __new_bucket = _M_bkt_num(__first->_M_val, __n); 931 _M_buckets[__bucket] = __first->_M_next; 932 __first->_M_next = __tmp[__new_bucket]; 933 __tmp[__new_bucket] = __first; 934 __first = _M_buckets[__bucket]; 935 } 936 } 937 _M_buckets.swap(__tmp); 938 } 939 # ifdef __STL_USE_EXCEPTIONS 940 catch(...) { 941 for (size_type __bucket = 0; __bucket < __tmp.size(); ++__bucket) { 942 while (__tmp[__bucket]) { 943 _Node* __next = __tmp[__bucket]->_M_next; 944 _M_delete_node(__tmp[__bucket]); 945 __tmp[__bucket] = __next; 946 } 947 } 948 throw; 949 } 950 # endif /* __STL_USE_EXCEPTIONS */ 951 } 952 } 953 } 954 955 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 956 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 957 ::_M_erase_bucket(const size_type __n, _Node* __first, _Node* __last) 958 { 959 _Node* __cur = _M_buckets[__n]; 960 if (__cur == __first) 961 _M_erase_bucket(__n, __last); 962 else { 963 _Node* __next; 964 for (__next = __cur->_M_next; 965 __next != __first; 966 __cur = __next, __next = __cur->_M_next) 967 ; 968 while (__next) { 969 __cur->_M_next = __next->_M_next; 970 _M_delete_node(__next); 971 __next = __cur->_M_next; 972 --_M_num_elements; 973 } 974 } 975 } 976 977 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 978 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 979 ::_M_erase_bucket(const size_type __n, _Node* __last) 980 { 981 _Node* __cur = _M_buckets[__n]; 982 while (__cur != __last) { 983 _Node* __next = __cur->_M_next; 984 _M_delete_node(__cur); 985 __cur = __next; 986 _M_buckets[__n] = __cur; 987 --_M_num_elements; 988 } 989 } 990 991 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 992 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All>::clear() 993 { 994 for (size_type __i = 0; __i < _M_buckets.size(); ++__i) { 995 _Node* __cur = _M_buckets[__i]; 996 while (__cur != 0) { 997 _Node* __next = __cur->_M_next; 998 _M_delete_node(__cur); 999 __cur = __next; 1000 } 1001 _M_buckets[__i] = 0; 1002 } 1003 _M_num_elements = 0; 1004 } 1005 1006 1007 template <class _Val, class _Key, class _HF, class _Ex, class _Eq, class _All> 1008 void hashtable<_Val,_Key,_HF,_Ex,_Eq,_All> 1009 ::_M_copy_from(const hashtable& __ht) 1010 { 1011 _M_buckets.clear(); 1012 _M_buckets.reserve(__ht._M_buckets.size()); 1013 _M_buckets.insert(_M_buckets.end(), __ht._M_buckets.size(), (_Node*) 0); 1014 __STL_TRY { 1015 for (size_type __i = 0; __i < __ht._M_buckets.size(); ++__i) { 1016 if (const _Node* __cur = __ht._M_buckets[__i]) { 1017 _Node* ___copy = _M_new_node(__cur->_M_val); 1018 _M_buckets[__i] = ___copy; 1019 1020 for (_Node* __next = __cur->_M_next; 1021 __next; 1022 __cur = __next, __next = __cur->_M_next) { 1023 ___copy->_M_next = _M_new_node(__next->_M_val); 1024 ___copy = ___copy->_M_next; 1025 } 1026 } 1027 } 1028 _M_num_elements = __ht._M_num_elements; 1029 } 1030 __STL_UNWIND(clear()); 1031 } 1032 1033 __STL_END_NAMESPACE 1034 1035 #endif /* __SGI_STL_INTERNAL_HASHTABLE_H */ 1036 1037 // Local Variables: 1038 // mode:C++ 1039 // End: 1040