1 ///////////////////////////////////////////////////////////////////////////// 2 // 3 // (C) Copyright Olaf Krzikalla 2004-2006. 4 // (C) Copyright Ion Gaztanaga 2006-2014 5 // 6 // Distributed under the Boost Software License, Version 1.0. 7 // (See accompanying file LICENSE_1_0.txt or copy at 8 // http://www.boost.org/LICENSE_1_0.txt) 9 // 10 // See http://www.boost.org/libs/intrusive for documentation. 11 // 12 ///////////////////////////////////////////////////////////////////////////// 13 #ifndef BOOST_INTRUSIVE_SET_HPP 14 #define BOOST_INTRUSIVE_SET_HPP 15 16 #include <boost/intrusive/detail/config_begin.hpp> 17 #include <boost/intrusive/intrusive_fwd.hpp> 18 19 #include <boost/intrusive/detail/mpl.hpp> 20 #include <boost/intrusive/rbtree.hpp> 21 #include <boost/move/utility_core.hpp> 22 #include <boost/static_assert.hpp> 23 24 #if defined(BOOST_HAS_PRAGMA_ONCE) 25 # pragma once 26 #endif 27 28 namespace boost { 29 namespace intrusive { 30 31 #if !defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 32 template<class ValueTraits, class VoidOrKeyOfValue, class Compare, class SizeType, bool ConstantTimeSize, typename HeaderHolder> 33 class multiset_impl; 34 #endif 35 36 //! The class template set is an intrusive container, that mimics most of 37 //! the interface of std::set as described in the C++ standard. 38 //! 39 //! The template parameter \c T is the type to be managed by the container. 40 //! The user can specify additional options and if no options are provided 41 //! default options are used. 42 //! 43 //! The container supports the following options: 44 //! \c base_hook<>/member_hook<>/value_traits<>, 45 //! \c constant_time_size<>, \c size_type<> and 46 //! \c compare<>. 47 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 48 template<class T, class ...Options> 49 #else 50 template<class ValueTraits, class VoidOrKeyOfValue, class Compare, class SizeType, bool ConstantTimeSize, typename HeaderHolder> 51 #endif 52 class set_impl 53 #ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED 54 : public bstree_impl<ValueTraits, VoidOrKeyOfValue, Compare, SizeType, ConstantTimeSize, RbTreeAlgorithms, HeaderHolder> 55 #endif 56 { 57 /// @cond 58 typedef bstree_impl<ValueTraits, VoidOrKeyOfValue, Compare, SizeType, ConstantTimeSize, RbTreeAlgorithms, HeaderHolder> tree_type; 59 BOOST_MOVABLE_BUT_NOT_COPYABLE(set_impl) 60 61 typedef tree_type implementation_defined; 62 /// @endcond 63 64 public: 65 typedef typename implementation_defined::value_type value_type; 66 typedef typename implementation_defined::key_type key_type; 67 typedef typename implementation_defined::key_of_value key_of_value; 68 typedef typename implementation_defined::value_traits value_traits; 69 typedef typename implementation_defined::pointer pointer; 70 typedef typename implementation_defined::const_pointer const_pointer; 71 typedef typename implementation_defined::reference reference; 72 typedef typename implementation_defined::const_reference const_reference; 73 typedef typename implementation_defined::difference_type difference_type; 74 typedef typename implementation_defined::size_type size_type; 75 typedef typename implementation_defined::value_compare value_compare; 76 typedef typename implementation_defined::key_compare key_compare; 77 typedef typename implementation_defined::iterator iterator; 78 typedef typename implementation_defined::const_iterator const_iterator; 79 typedef typename implementation_defined::reverse_iterator reverse_iterator; 80 typedef typename implementation_defined::const_reverse_iterator const_reverse_iterator; 81 typedef typename implementation_defined::insert_commit_data insert_commit_data; 82 typedef typename implementation_defined::node_traits node_traits; 83 typedef typename implementation_defined::node node; 84 typedef typename implementation_defined::node_ptr node_ptr; 85 typedef typename implementation_defined::const_node_ptr const_node_ptr; 86 typedef typename implementation_defined::node_algorithms node_algorithms; 87 88 static const bool constant_time_size = tree_type::constant_time_size; 89 90 public: 91 //! @copydoc ::boost::intrusive::rbtree::rbtree() set_impl()92 set_impl() 93 : tree_type() 94 {} 95 96 //! @copydoc ::boost::intrusive::rbtree::rbtree(const key_compare &,const value_traits &) set_impl(const key_compare & cmp,const value_traits & v_traits=value_traits ())97 explicit set_impl( const key_compare &cmp, const value_traits &v_traits = value_traits()) 98 : tree_type(cmp, v_traits) 99 {} 100 101 //! @copydoc ::boost::intrusive::rbtree::rbtree(bool,Iterator,Iterator,const key_compare &,const value_traits &) 102 template<class Iterator> set_impl(Iterator b,Iterator e,const key_compare & cmp=key_compare (),const value_traits & v_traits=value_traits ())103 set_impl( Iterator b, Iterator e 104 , const key_compare &cmp = key_compare() 105 , const value_traits &v_traits = value_traits()) 106 : tree_type(true, b, e, cmp, v_traits) 107 {} 108 109 //! @copydoc ::boost::intrusive::rbtree::rbtree(rbtree &&) set_impl(BOOST_RV_REF (set_impl)x)110 set_impl(BOOST_RV_REF(set_impl) x) 111 : tree_type(BOOST_MOVE_BASE(tree_type, x)) 112 {} 113 114 //! @copydoc ::boost::intrusive::rbtree::operator=(rbtree &&) operator =(BOOST_RV_REF (set_impl)x)115 set_impl& operator=(BOOST_RV_REF(set_impl) x) 116 { return static_cast<set_impl&>(tree_type::operator=(BOOST_MOVE_BASE(tree_type, x))); } 117 118 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 119 //! @copydoc ::boost::intrusive::rbtree::~rbtree() 120 ~set_impl(); 121 122 //! @copydoc ::boost::intrusive::rbtree::begin() 123 iterator begin(); 124 125 //! @copydoc ::boost::intrusive::rbtree::begin()const 126 const_iterator begin() const; 127 128 //! @copydoc ::boost::intrusive::rbtree::cbegin()const 129 const_iterator cbegin() const; 130 131 //! @copydoc ::boost::intrusive::rbtree::end() 132 iterator end(); 133 134 //! @copydoc ::boost::intrusive::rbtree::end()const 135 const_iterator end() const; 136 137 //! @copydoc ::boost::intrusive::rbtree::cend()const 138 const_iterator cend() const; 139 140 //! @copydoc ::boost::intrusive::rbtree::rbegin() 141 reverse_iterator rbegin(); 142 143 //! @copydoc ::boost::intrusive::rbtree::rbegin()const 144 const_reverse_iterator rbegin() const; 145 146 //! @copydoc ::boost::intrusive::rbtree::crbegin()const 147 const_reverse_iterator crbegin() const; 148 149 //! @copydoc ::boost::intrusive::rbtree::rend() 150 reverse_iterator rend(); 151 152 //! @copydoc ::boost::intrusive::rbtree::rend()const 153 const_reverse_iterator rend() const; 154 155 //! @copydoc ::boost::intrusive::rbtree::crend()const 156 const_reverse_iterator crend() const; 157 158 //! @copydoc ::boost::intrusive::rbtree::root() 159 iterator root(); 160 161 //! @copydoc ::boost::intrusive::rbtree::root()const 162 const_iterator root() const; 163 164 //! @copydoc ::boost::intrusive::rbtree::croot()const 165 const_iterator croot() const; 166 167 //! @copydoc ::boost::intrusive::rbtree::container_from_end_iterator(iterator) 168 static set_impl &container_from_end_iterator(iterator end_iterator); 169 170 //! @copydoc ::boost::intrusive::rbtree::container_from_end_iterator(const_iterator) 171 static const set_impl &container_from_end_iterator(const_iterator end_iterator); 172 173 //! @copydoc ::boost::intrusive::rbtree::container_from_iterator(iterator) 174 static set_impl &container_from_iterator(iterator it); 175 176 //! @copydoc ::boost::intrusive::rbtree::container_from_iterator(const_iterator) 177 static const set_impl &container_from_iterator(const_iterator it); 178 179 //! @copydoc ::boost::intrusive::rbtree::key_comp()const 180 key_compare key_comp() const; 181 182 //! @copydoc ::boost::intrusive::rbtree::value_comp()const 183 value_compare value_comp() const; 184 185 //! @copydoc ::boost::intrusive::rbtree::empty()const 186 bool empty() const; 187 188 //! @copydoc ::boost::intrusive::rbtree::size()const 189 size_type size() const; 190 191 //! @copydoc ::boost::intrusive::rbtree::swap 192 void swap(set_impl& other); 193 194 //! @copydoc ::boost::intrusive::rbtree::clone_from(const rbtree&,Cloner,Disposer) 195 template <class Cloner, class Disposer> 196 void clone_from(const set_impl &src, Cloner cloner, Disposer disposer); 197 198 #else 199 200 using tree_type::clone_from; 201 202 #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 203 204 //! @copydoc ::boost::intrusive::rbtree::clone_from(rbtree&&,Cloner,Disposer) 205 template <class Cloner, class Disposer> clone_from(BOOST_RV_REF (set_impl)src,Cloner cloner,Disposer disposer)206 void clone_from(BOOST_RV_REF(set_impl) src, Cloner cloner, Disposer disposer) 207 { tree_type::clone_from(BOOST_MOVE_BASE(tree_type, src), cloner, disposer); } 208 209 //! @copydoc ::boost::intrusive::rbtree::insert_unique(reference) insert(reference value)210 std::pair<iterator, bool> insert(reference value) 211 { return tree_type::insert_unique(value); } 212 213 //! @copydoc ::boost::intrusive::rbtree::insert_unique(const_iterator,reference) insert(const_iterator hint,reference value)214 iterator insert(const_iterator hint, reference value) 215 { return tree_type::insert_unique(hint, value); } 216 217 //! @copydoc ::boost::intrusive::rbtree::insert_unique_check(const key_type&,insert_commit_data&) insert_check(const key_type & key,insert_commit_data & commit_data)218 std::pair<iterator, bool> insert_check 219 (const key_type &key, insert_commit_data &commit_data) 220 { return tree_type::insert_unique_check(key, commit_data); } 221 222 //! @copydoc ::boost::intrusive::rbtree::insert_unique_check(const_iterator,const key_type&,insert_commit_data&) insert_check(const_iterator hint,const key_type & key,insert_commit_data & commit_data)223 std::pair<iterator, bool> insert_check 224 (const_iterator hint, const key_type &key 225 ,insert_commit_data &commit_data) 226 { return tree_type::insert_unique_check(hint, key, commit_data); } 227 228 //! @copydoc ::boost::intrusive::rbtree::insert_unique_check(const KeyType&,KeyTypeKeyCompare,insert_commit_data&) 229 template<class KeyType, class KeyTypeKeyCompare> insert_check(const KeyType & key,KeyTypeKeyCompare comp,insert_commit_data & commit_data)230 std::pair<iterator, bool> insert_check 231 (const KeyType &key, KeyTypeKeyCompare comp, insert_commit_data &commit_data) 232 { return tree_type::insert_unique_check(key, comp, commit_data); } 233 234 //! @copydoc ::boost::intrusive::rbtree::insert_unique_check(const_iterator,const KeyType&,KeyTypeKeyCompare,insert_commit_data&) 235 template<class KeyType, class KeyTypeKeyCompare> insert_check(const_iterator hint,const KeyType & key,KeyTypeKeyCompare comp,insert_commit_data & commit_data)236 std::pair<iterator, bool> insert_check 237 (const_iterator hint, const KeyType &key 238 ,KeyTypeKeyCompare comp, insert_commit_data &commit_data) 239 { return tree_type::insert_unique_check(hint, key, comp, commit_data); } 240 241 //! @copydoc ::boost::intrusive::rbtree::insert_unique(Iterator,Iterator) 242 template<class Iterator> insert(Iterator b,Iterator e)243 void insert(Iterator b, Iterator e) 244 { tree_type::insert_unique(b, e); } 245 246 //! @copydoc ::boost::intrusive::rbtree::insert_unique_commit insert_commit(reference value,const insert_commit_data & commit_data)247 iterator insert_commit(reference value, const insert_commit_data &commit_data) 248 { return tree_type::insert_unique_commit(value, commit_data); } 249 250 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 251 //! @copydoc ::boost::intrusive::rbtree::insert_before 252 iterator insert_before(const_iterator pos, reference value); 253 254 //! @copydoc ::boost::intrusive::rbtree::push_back 255 void push_back(reference value); 256 257 //! @copydoc ::boost::intrusive::rbtree::push_front 258 void push_front(reference value); 259 260 //! @copydoc ::boost::intrusive::rbtree::erase(const_iterator) 261 iterator erase(const_iterator i); 262 263 //! @copydoc ::boost::intrusive::rbtree::erase(const_iterator,const_iterator) 264 iterator erase(const_iterator b, const_iterator e); 265 266 //! @copydoc ::boost::intrusive::rbtree::erase(const key_type &) 267 size_type erase(const key_type &key); 268 269 //! @copydoc ::boost::intrusive::rbtree::erase(const KeyType&,KeyTypeKeyCompare) 270 template<class KeyType, class KeyTypeKeyCompare> 271 size_type erase(const KeyType& key, KeyTypeKeyCompare comp); 272 273 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const_iterator,Disposer) 274 template<class Disposer> 275 iterator erase_and_dispose(const_iterator i, Disposer disposer); 276 277 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const_iterator,const_iterator,Disposer) 278 template<class Disposer> 279 iterator erase_and_dispose(const_iterator b, const_iterator e, Disposer disposer); 280 281 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const key_type &, Disposer) 282 template<class Disposer> 283 size_type erase_and_dispose(const key_type &key, Disposer disposer); 284 285 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const KeyType&,KeyTypeKeyCompare,Disposer) 286 template<class KeyType, class KeyTypeKeyCompare, class Disposer> 287 size_type erase_and_dispose(const KeyType& key, KeyTypeKeyCompare comp, Disposer disposer); 288 289 //! @copydoc ::boost::intrusive::rbtree::clear 290 void clear(); 291 292 //! @copydoc ::boost::intrusive::rbtree::clear_and_dispose 293 template<class Disposer> 294 void clear_and_dispose(Disposer disposer); 295 296 #endif // #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 297 298 //! @copydoc ::boost::intrusive::rbtree::count(const key_type &)const count(const key_type & key) const299 size_type count(const key_type &key) const 300 { return static_cast<size_type>(this->tree_type::find(key) != this->tree_type::cend()); } 301 302 //! @copydoc ::boost::intrusive::rbtree::count(const KeyType&,KeyTypeKeyCompare)const 303 template<class KeyType, class KeyTypeKeyCompare> count(const KeyType & key,KeyTypeKeyCompare comp) const304 size_type count(const KeyType& key, KeyTypeKeyCompare comp) const 305 { return static_cast<size_type>(this->tree_type::find(key, comp) != this->tree_type::cend()); } 306 307 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 308 309 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const key_type &) 310 iterator lower_bound(const key_type &key); 311 312 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const KeyType&,KeyTypeKeyCompare) 313 template<class KeyType, class KeyTypeKeyCompare> 314 iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp); 315 316 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const key_type &)const 317 const_iterator lower_bound(const key_type &key) const; 318 319 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const KeyType&,KeyTypeKeyCompare)const 320 template<class KeyType, class KeyTypeKeyCompare> 321 const_iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp) const; 322 323 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const key_type &) 324 iterator upper_bound(const key_type &key); 325 326 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const KeyType&,KeyTypeKeyCompare) 327 template<class KeyType, class KeyTypeKeyCompare> 328 iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp); 329 330 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const key_type &)const 331 const_iterator upper_bound(const key_type &key) const; 332 333 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const KeyType&,KeyTypeKeyCompare)const 334 template<class KeyType, class KeyTypeKeyCompare> 335 const_iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp) const; 336 337 //! @copydoc ::boost::intrusive::rbtree::find(const key_type &) 338 iterator find(const key_type &key); 339 340 //! @copydoc ::boost::intrusive::rbtree::find(const KeyType&,KeyTypeKeyCompare) 341 template<class KeyType, class KeyTypeKeyCompare> 342 iterator find(const KeyType& key, KeyTypeKeyCompare comp); 343 344 //! @copydoc ::boost::intrusive::rbtree::find(const key_type &)const 345 const_iterator find(const key_type &key) const; 346 347 //! @copydoc ::boost::intrusive::rbtree::find(const KeyType&,KeyTypeKeyCompare)const 348 template<class KeyType, class KeyTypeKeyCompare> 349 const_iterator find(const KeyType& key, KeyTypeKeyCompare comp) const; 350 351 #endif // #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 352 353 //! @copydoc ::boost::intrusive::rbtree::equal_range(const key_type &) equal_range(const key_type & key)354 std::pair<iterator,iterator> equal_range(const key_type &key) 355 { return this->tree_type::lower_bound_range(key); } 356 357 //! @copydoc ::boost::intrusive::rbtree::equal_range(const KeyType&,KeyTypeKeyCompare) 358 template<class KeyType, class KeyTypeKeyCompare> equal_range(const KeyType & key,KeyTypeKeyCompare comp)359 std::pair<iterator,iterator> equal_range(const KeyType& key, KeyTypeKeyCompare comp) 360 { return this->tree_type::equal_range(key, comp); } 361 362 //! @copydoc ::boost::intrusive::rbtree::equal_range(const key_type &)const 363 std::pair<const_iterator, const_iterator> equal_range(const key_type & key) const364 equal_range(const key_type &key) const 365 { return this->tree_type::lower_bound_range(key); } 366 367 //! @copydoc ::boost::intrusive::rbtree::equal_range(const KeyType&,KeyTypeKeyCompare)const 368 template<class KeyType, class KeyTypeKeyCompare> 369 std::pair<const_iterator, const_iterator> equal_range(const KeyType & key,KeyTypeKeyCompare comp) const370 equal_range(const KeyType& key, KeyTypeKeyCompare comp) const 371 { return this->tree_type::equal_range(key, comp); } 372 373 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 374 375 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const key_type &,const key_type &,bool,bool) 376 std::pair<iterator,iterator> bounded_range 377 (const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed); 378 379 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool) 380 template<class KeyType, class KeyTypeKeyCompare> 381 std::pair<iterator,iterator> bounded_range 382 (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed); 383 384 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const key_type &,const key_type &,bool,bool)const 385 std::pair<const_iterator, const_iterator> 386 bounded_range(const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed) const; 387 388 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool)const 389 template<class KeyType, class KeyTypeKeyCompare> 390 std::pair<const_iterator, const_iterator> bounded_range 391 (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed) const; 392 393 //! @copydoc ::boost::intrusive::rbtree::s_iterator_to(reference) 394 static iterator s_iterator_to(reference value); 395 396 //! @copydoc ::boost::intrusive::rbtree::s_iterator_to(const_reference) 397 static const_iterator s_iterator_to(const_reference value); 398 399 //! @copydoc ::boost::intrusive::rbtree::iterator_to(reference) 400 iterator iterator_to(reference value); 401 402 //! @copydoc ::boost::intrusive::rbtree::iterator_to(const_reference)const 403 const_iterator iterator_to(const_reference value) const; 404 405 //! @copydoc ::boost::intrusive::rbtree::init_node(reference) 406 static void init_node(reference value); 407 408 //! @copydoc ::boost::intrusive::rbtree::unlink_leftmost_without_rebalance 409 pointer unlink_leftmost_without_rebalance(); 410 411 //! @copydoc ::boost::intrusive::rbtree::replace_node 412 void replace_node(iterator replace_this, reference with_this); 413 414 //! @copydoc ::boost::intrusive::rbtree::remove_node 415 void remove_node(reference value); 416 417 //! @copydoc ::boost::intrusive::rbtree::merge_unique 418 template<class ...Options2> 419 void merge(set<T, Options2...> &source); 420 421 //! @copydoc ::boost::intrusive::rbtree::merge_unique 422 template<class ...Options2> 423 void merge(multiset<T, Options2...> &source); 424 425 #else 426 427 template<class Compare2> merge(set_impl<ValueTraits,VoidOrKeyOfValue,Compare2,SizeType,ConstantTimeSize,HeaderHolder> & source)428 void merge(set_impl<ValueTraits, VoidOrKeyOfValue, Compare2, SizeType, ConstantTimeSize, HeaderHolder> &source) 429 { return tree_type::merge_unique(source); } 430 431 432 template<class Compare2> merge(multiset_impl<ValueTraits,VoidOrKeyOfValue,Compare2,SizeType,ConstantTimeSize,HeaderHolder> & source)433 void merge(multiset_impl<ValueTraits, VoidOrKeyOfValue, Compare2, SizeType, ConstantTimeSize, HeaderHolder> &source) 434 { return tree_type::merge_unique(source); } 435 436 #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 437 }; 438 439 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 440 441 template<class T, class ...Options> 442 bool operator!= (const set_impl<T, Options...> &x, const set_impl<T, Options...> &y); 443 444 template<class T, class ...Options> 445 bool operator>(const set_impl<T, Options...> &x, const set_impl<T, Options...> &y); 446 447 template<class T, class ...Options> 448 bool operator<=(const set_impl<T, Options...> &x, const set_impl<T, Options...> &y); 449 450 template<class T, class ...Options> 451 bool operator>=(const set_impl<T, Options...> &x, const set_impl<T, Options...> &y); 452 453 template<class T, class ...Options> 454 void swap(set_impl<T, Options...> &x, set_impl<T, Options...> &y); 455 456 #endif //#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 457 458 //! Helper metafunction to define a \c set that yields to the same type when the 459 //! same options (either explicitly or implicitly) are used. 460 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 461 template<class T, class ...Options> 462 #else 463 template<class T, class O1 = void, class O2 = void 464 , class O3 = void, class O4 = void 465 , class O5 = void, class O6 = void> 466 #endif 467 struct make_set 468 { 469 /// @cond 470 typedef typename pack_options 471 < rbtree_defaults, 472 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 473 O1, O2, O3, O4, O5, O6 474 #else 475 Options... 476 #endif 477 >::type packed_options; 478 479 typedef typename detail::get_value_traits 480 <T, typename packed_options::proto_value_traits>::type value_traits; 481 482 typedef set_impl 483 < value_traits 484 , typename packed_options::key_of_value 485 , typename packed_options::compare 486 , typename packed_options::size_type 487 , packed_options::constant_time_size 488 , typename packed_options::header_holder_type 489 > implementation_defined; 490 /// @endcond 491 typedef implementation_defined type; 492 }; 493 494 #ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED 495 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 496 template<class T, class O1, class O2, class O3, class O4, class O5, class O6> 497 #else 498 template<class T, class ...Options> 499 #endif 500 class set 501 : public make_set<T, 502 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 503 O1, O2, O3, O4, O5, O6 504 #else 505 Options... 506 #endif 507 >::type 508 { 509 typedef typename make_set 510 <T, 511 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 512 O1, O2, O3, O4, O5, O6 513 #else 514 Options... 515 #endif 516 >::type Base; 517 518 BOOST_MOVABLE_BUT_NOT_COPYABLE(set) 519 public: 520 typedef typename Base::key_compare key_compare; 521 typedef typename Base::value_traits value_traits; 522 typedef typename Base::iterator iterator; 523 typedef typename Base::const_iterator const_iterator; 524 525 //Assert if passed value traits are compatible with the type 526 BOOST_STATIC_ASSERT((detail::is_same<typename value_traits::value_type, T>::value)); 527 set()528 BOOST_INTRUSIVE_FORCEINLINE set() 529 : Base() 530 {} 531 set(const key_compare & cmp,const value_traits & v_traits=value_traits ())532 BOOST_INTRUSIVE_FORCEINLINE explicit set( const key_compare &cmp, const value_traits &v_traits = value_traits()) 533 : Base(cmp, v_traits) 534 {} 535 536 template<class Iterator> set(Iterator b,Iterator e,const key_compare & cmp=key_compare (),const value_traits & v_traits=value_traits ())537 BOOST_INTRUSIVE_FORCEINLINE set( Iterator b, Iterator e 538 , const key_compare &cmp = key_compare() 539 , const value_traits &v_traits = value_traits()) 540 : Base(b, e, cmp, v_traits) 541 {} 542 set(BOOST_RV_REF (set)x)543 BOOST_INTRUSIVE_FORCEINLINE set(BOOST_RV_REF(set) x) 544 : Base(BOOST_MOVE_BASE(Base, x)) 545 {} 546 operator =(BOOST_RV_REF (set)x)547 BOOST_INTRUSIVE_FORCEINLINE set& operator=(BOOST_RV_REF(set) x) 548 { return static_cast<set &>(this->Base::operator=(BOOST_MOVE_BASE(Base, x))); } 549 550 template <class Cloner, class Disposer> clone_from(const set & src,Cloner cloner,Disposer disposer)551 BOOST_INTRUSIVE_FORCEINLINE void clone_from(const set &src, Cloner cloner, Disposer disposer) 552 { Base::clone_from(src, cloner, disposer); } 553 554 template <class Cloner, class Disposer> clone_from(BOOST_RV_REF (set)src,Cloner cloner,Disposer disposer)555 BOOST_INTRUSIVE_FORCEINLINE void clone_from(BOOST_RV_REF(set) src, Cloner cloner, Disposer disposer) 556 { Base::clone_from(BOOST_MOVE_BASE(Base, src), cloner, disposer); } 557 container_from_end_iterator(iterator end_iterator)558 BOOST_INTRUSIVE_FORCEINLINE static set &container_from_end_iterator(iterator end_iterator) 559 { return static_cast<set &>(Base::container_from_end_iterator(end_iterator)); } 560 container_from_end_iterator(const_iterator end_iterator)561 BOOST_INTRUSIVE_FORCEINLINE static const set &container_from_end_iterator(const_iterator end_iterator) 562 { return static_cast<const set &>(Base::container_from_end_iterator(end_iterator)); } 563 container_from_iterator(iterator it)564 BOOST_INTRUSIVE_FORCEINLINE static set &container_from_iterator(iterator it) 565 { return static_cast<set &>(Base::container_from_iterator(it)); } 566 container_from_iterator(const_iterator it)567 BOOST_INTRUSIVE_FORCEINLINE static const set &container_from_iterator(const_iterator it) 568 { return static_cast<const set &>(Base::container_from_iterator(it)); } 569 }; 570 571 #endif 572 573 //! The class template multiset is an intrusive container, that mimics most of 574 //! the interface of std::multiset as described in the C++ standard. 575 //! 576 //! The template parameter \c T is the type to be managed by the container. 577 //! The user can specify additional options and if no options are provided 578 //! default options are used. 579 //! 580 //! The container supports the following options: 581 //! \c base_hook<>/member_hook<>/value_traits<>, 582 //! \c constant_time_size<>, \c size_type<> and 583 //! \c compare<>. 584 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 585 template<class T, class ...Options> 586 #else 587 template<class ValueTraits, class VoidOrKeyOfValue, class Compare, class SizeType, bool ConstantTimeSize, typename HeaderHolder> 588 #endif 589 class multiset_impl 590 #ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED 591 : public bstree_impl<ValueTraits, VoidOrKeyOfValue, Compare, SizeType, ConstantTimeSize, RbTreeAlgorithms, HeaderHolder> 592 #endif 593 { 594 /// @cond 595 typedef bstree_impl<ValueTraits, VoidOrKeyOfValue, Compare, SizeType, ConstantTimeSize, RbTreeAlgorithms, HeaderHolder> tree_type; 596 597 BOOST_MOVABLE_BUT_NOT_COPYABLE(multiset_impl) 598 typedef tree_type implementation_defined; 599 /// @endcond 600 601 public: 602 typedef typename implementation_defined::value_type value_type; 603 typedef typename implementation_defined::key_type key_type; 604 typedef typename implementation_defined::key_of_value key_of_value; 605 typedef typename implementation_defined::value_traits value_traits; 606 typedef typename implementation_defined::pointer pointer; 607 typedef typename implementation_defined::const_pointer const_pointer; 608 typedef typename implementation_defined::reference reference; 609 typedef typename implementation_defined::const_reference const_reference; 610 typedef typename implementation_defined::difference_type difference_type; 611 typedef typename implementation_defined::size_type size_type; 612 typedef typename implementation_defined::value_compare value_compare; 613 typedef typename implementation_defined::key_compare key_compare; 614 typedef typename implementation_defined::iterator iterator; 615 typedef typename implementation_defined::const_iterator const_iterator; 616 typedef typename implementation_defined::reverse_iterator reverse_iterator; 617 typedef typename implementation_defined::const_reverse_iterator const_reverse_iterator; 618 typedef typename implementation_defined::insert_commit_data insert_commit_data; 619 typedef typename implementation_defined::node_traits node_traits; 620 typedef typename implementation_defined::node node; 621 typedef typename implementation_defined::node_ptr node_ptr; 622 typedef typename implementation_defined::const_node_ptr const_node_ptr; 623 typedef typename implementation_defined::node_algorithms node_algorithms; 624 625 static const bool constant_time_size = tree_type::constant_time_size; 626 627 public: 628 //! @copydoc ::boost::intrusive::rbtree::rbtree() multiset_impl()629 multiset_impl() 630 : tree_type() 631 {} 632 633 //! @copydoc ::boost::intrusive::rbtree::rbtree(const key_compare &,const value_traits &) multiset_impl(const key_compare & cmp,const value_traits & v_traits=value_traits ())634 explicit multiset_impl( const key_compare &cmp, const value_traits &v_traits = value_traits()) 635 : tree_type(cmp, v_traits) 636 {} 637 638 //! @copydoc ::boost::intrusive::rbtree::rbtree(bool,Iterator,Iterator,const key_compare &,const value_traits &) 639 template<class Iterator> multiset_impl(Iterator b,Iterator e,const key_compare & cmp=key_compare (),const value_traits & v_traits=value_traits ())640 multiset_impl( Iterator b, Iterator e 641 , const key_compare &cmp = key_compare() 642 , const value_traits &v_traits = value_traits()) 643 : tree_type(false, b, e, cmp, v_traits) 644 {} 645 646 //! @copydoc ::boost::intrusive::rbtree::rbtree(rbtree &&) multiset_impl(BOOST_RV_REF (multiset_impl)x)647 multiset_impl(BOOST_RV_REF(multiset_impl) x) 648 : tree_type(BOOST_MOVE_BASE(tree_type, x)) 649 {} 650 651 //! @copydoc ::boost::intrusive::rbtree::operator=(rbtree &&) operator =(BOOST_RV_REF (multiset_impl)x)652 multiset_impl& operator=(BOOST_RV_REF(multiset_impl) x) 653 { return static_cast<multiset_impl&>(tree_type::operator=(BOOST_MOVE_BASE(tree_type, x))); } 654 655 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 656 //! @copydoc ::boost::intrusive::rbtree::~rbtree() 657 ~multiset_impl(); 658 659 //! @copydoc ::boost::intrusive::rbtree::begin() 660 iterator begin(); 661 662 //! @copydoc ::boost::intrusive::rbtree::begin()const 663 const_iterator begin() const; 664 665 //! @copydoc ::boost::intrusive::rbtree::cbegin()const 666 const_iterator cbegin() const; 667 668 //! @copydoc ::boost::intrusive::rbtree::end() 669 iterator end(); 670 671 //! @copydoc ::boost::intrusive::rbtree::end()const 672 const_iterator end() const; 673 674 //! @copydoc ::boost::intrusive::rbtree::cend()const 675 const_iterator cend() const; 676 677 //! @copydoc ::boost::intrusive::rbtree::rbegin() 678 reverse_iterator rbegin(); 679 680 //! @copydoc ::boost::intrusive::rbtree::rbegin()const 681 const_reverse_iterator rbegin() const; 682 683 //! @copydoc ::boost::intrusive::rbtree::crbegin()const 684 const_reverse_iterator crbegin() const; 685 686 //! @copydoc ::boost::intrusive::rbtree::rend() 687 reverse_iterator rend(); 688 689 //! @copydoc ::boost::intrusive::rbtree::rend()const 690 const_reverse_iterator rend() const; 691 692 //! @copydoc ::boost::intrusive::rbtree::crend()const 693 const_reverse_iterator crend() const; 694 695 //! @copydoc ::boost::intrusive::rbtree::root() 696 iterator root(); 697 698 //! @copydoc ::boost::intrusive::rbtree::root()const 699 const_iterator root() const; 700 701 //! @copydoc ::boost::intrusive::rbtree::croot()const 702 const_iterator croot() const; 703 704 //! @copydoc ::boost::intrusive::rbtree::container_from_end_iterator(iterator) 705 static multiset_impl &container_from_end_iterator(iterator end_iterator); 706 707 //! @copydoc ::boost::intrusive::rbtree::container_from_end_iterator(const_iterator) 708 static const multiset_impl &container_from_end_iterator(const_iterator end_iterator); 709 710 //! @copydoc ::boost::intrusive::rbtree::container_from_iterator(iterator) 711 static multiset_impl &container_from_iterator(iterator it); 712 713 //! @copydoc ::boost::intrusive::rbtree::container_from_iterator(const_iterator) 714 static const multiset_impl &container_from_iterator(const_iterator it); 715 716 //! @copydoc ::boost::intrusive::rbtree::key_comp()const 717 key_compare key_comp() const; 718 719 //! @copydoc ::boost::intrusive::rbtree::value_comp()const 720 value_compare value_comp() const; 721 722 //! @copydoc ::boost::intrusive::rbtree::empty()const 723 bool empty() const; 724 725 //! @copydoc ::boost::intrusive::rbtree::size()const 726 size_type size() const; 727 728 //! @copydoc ::boost::intrusive::rbtree::swap 729 void swap(multiset_impl& other); 730 731 //! @copydoc ::boost::intrusive::rbtree::clone_from(const rbtree&,Cloner,Disposer) 732 template <class Cloner, class Disposer> 733 void clone_from(const multiset_impl &src, Cloner cloner, Disposer disposer); 734 735 #else 736 737 using tree_type::clone_from; 738 739 #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 740 741 //! @copydoc ::boost::intrusive::rbtree::clone_from(rbtree&&,Cloner,Disposer) 742 template <class Cloner, class Disposer> clone_from(BOOST_RV_REF (multiset_impl)src,Cloner cloner,Disposer disposer)743 void clone_from(BOOST_RV_REF(multiset_impl) src, Cloner cloner, Disposer disposer) 744 { tree_type::clone_from(BOOST_MOVE_BASE(tree_type, src), cloner, disposer); } 745 746 //! @copydoc ::boost::intrusive::rbtree::insert_equal(reference) insert(reference value)747 iterator insert(reference value) 748 { return tree_type::insert_equal(value); } 749 750 //! @copydoc ::boost::intrusive::rbtree::insert_equal(const_iterator,reference) insert(const_iterator hint,reference value)751 iterator insert(const_iterator hint, reference value) 752 { return tree_type::insert_equal(hint, value); } 753 754 //! @copydoc ::boost::intrusive::rbtree::insert_equal(Iterator,Iterator) 755 template<class Iterator> insert(Iterator b,Iterator e)756 void insert(Iterator b, Iterator e) 757 { tree_type::insert_equal(b, e); } 758 759 #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 760 //! @copydoc ::boost::intrusive::rbtree::insert_before 761 iterator insert_before(const_iterator pos, reference value); 762 763 //! @copydoc ::boost::intrusive::rbtree::push_back 764 void push_back(reference value); 765 766 //! @copydoc ::boost::intrusive::rbtree::push_front 767 void push_front(reference value); 768 769 //! @copydoc ::boost::intrusive::rbtree::erase(const_iterator) 770 iterator erase(const_iterator i); 771 772 //! @copydoc ::boost::intrusive::rbtree::erase(const_iterator,const_iterator) 773 iterator erase(const_iterator b, const_iterator e); 774 775 //! @copydoc ::boost::intrusive::rbtree::erase(const key_type &) 776 size_type erase(const key_type &key); 777 778 //! @copydoc ::boost::intrusive::rbtree::erase(const KeyType&,KeyTypeKeyCompare) 779 template<class KeyType, class KeyTypeKeyCompare> 780 size_type erase(const KeyType& key, KeyTypeKeyCompare comp); 781 782 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const_iterator,Disposer) 783 template<class Disposer> 784 iterator erase_and_dispose(const_iterator i, Disposer disposer); 785 786 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const_iterator,const_iterator,Disposer) 787 template<class Disposer> 788 iterator erase_and_dispose(const_iterator b, const_iterator e, Disposer disposer); 789 790 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const key_type &, Disposer) 791 template<class Disposer> 792 size_type erase_and_dispose(const key_type &key, Disposer disposer); 793 794 //! @copydoc ::boost::intrusive::rbtree::erase_and_dispose(const KeyType&,KeyTypeKeyCompare,Disposer) 795 template<class KeyType, class KeyTypeKeyCompare, class Disposer> 796 size_type erase_and_dispose(const KeyType& key, KeyTypeKeyCompare comp, Disposer disposer); 797 798 //! @copydoc ::boost::intrusive::rbtree::clear 799 void clear(); 800 801 //! @copydoc ::boost::intrusive::rbtree::clear_and_dispose 802 template<class Disposer> 803 void clear_and_dispose(Disposer disposer); 804 805 //! @copydoc ::boost::intrusive::rbtree::count(const key_type &)const 806 size_type count(const key_type &key) const; 807 808 //! @copydoc ::boost::intrusive::rbtree::count(const KeyType&,KeyTypeKeyCompare)const 809 template<class KeyType, class KeyTypeKeyCompare> 810 size_type count(const KeyType& key, KeyTypeKeyCompare comp) const; 811 812 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const key_type &) 813 iterator lower_bound(const key_type &key); 814 815 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const KeyType&,KeyTypeKeyCompare) 816 template<class KeyType, class KeyTypeKeyCompare> 817 iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp); 818 819 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const key_type &)const 820 const_iterator lower_bound(const key_type &key) const; 821 822 //! @copydoc ::boost::intrusive::rbtree::lower_bound(const KeyType&,KeyTypeKeyCompare)const 823 template<class KeyType, class KeyTypeKeyCompare> 824 const_iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp) const; 825 826 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const key_type &) 827 iterator upper_bound(const key_type &key); 828 829 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const KeyType&,KeyTypeKeyCompare) 830 template<class KeyType, class KeyTypeKeyCompare> 831 iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp); 832 833 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const key_type &)const 834 const_iterator upper_bound(const key_type &key) const; 835 836 //! @copydoc ::boost::intrusive::rbtree::upper_bound(const KeyType&,KeyTypeKeyCompare)const 837 template<class KeyType, class KeyTypeKeyCompare> 838 const_iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp) const; 839 840 //! @copydoc ::boost::intrusive::rbtree::find(const key_type &) 841 iterator find(const key_type &key); 842 843 //! @copydoc ::boost::intrusive::rbtree::find(const KeyType&,KeyTypeKeyCompare) 844 template<class KeyType, class KeyTypeKeyCompare> 845 iterator find(const KeyType& key, KeyTypeKeyCompare comp); 846 847 //! @copydoc ::boost::intrusive::rbtree::find(const key_type &)const 848 const_iterator find(const key_type &key) const; 849 850 //! @copydoc ::boost::intrusive::rbtree::find(const KeyType&,KeyTypeKeyCompare)const 851 template<class KeyType, class KeyTypeKeyCompare> 852 const_iterator find(const KeyType& key, KeyTypeKeyCompare comp) const; 853 854 //! @copydoc ::boost::intrusive::rbtree::equal_range(const key_type &) 855 std::pair<iterator,iterator> equal_range(const key_type &key); 856 857 //! @copydoc ::boost::intrusive::rbtree::equal_range(const KeyType&,KeyTypeKeyCompare) 858 template<class KeyType, class KeyTypeKeyCompare> 859 std::pair<iterator,iterator> equal_range(const KeyType& key, KeyTypeKeyCompare comp); 860 861 //! @copydoc ::boost::intrusive::rbtree::equal_range(const key_type &)const 862 std::pair<const_iterator, const_iterator> 863 equal_range(const key_type &key) const; 864 865 //! @copydoc ::boost::intrusive::rbtree::equal_range(const KeyType&,KeyTypeKeyCompare)const 866 template<class KeyType, class KeyTypeKeyCompare> 867 std::pair<const_iterator, const_iterator> 868 equal_range(const KeyType& key, KeyTypeKeyCompare comp) const; 869 870 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const key_type &,const key_type &,bool,bool) 871 std::pair<iterator,iterator> bounded_range 872 (const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed); 873 874 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool) 875 template<class KeyType, class KeyTypeKeyCompare> 876 std::pair<iterator,iterator> bounded_range 877 (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed); 878 879 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const key_type &,const key_type &,bool,bool)const 880 std::pair<const_iterator, const_iterator> 881 bounded_range(const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed) const; 882 883 //! @copydoc ::boost::intrusive::rbtree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool)const 884 template<class KeyType, class KeyTypeKeyCompare> 885 std::pair<const_iterator, const_iterator> bounded_range 886 (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed) const; 887 888 //! @copydoc ::boost::intrusive::rbtree::s_iterator_to(reference) 889 static iterator s_iterator_to(reference value); 890 891 //! @copydoc ::boost::intrusive::rbtree::s_iterator_to(const_reference) 892 static const_iterator s_iterator_to(const_reference value); 893 894 //! @copydoc ::boost::intrusive::rbtree::iterator_to(reference) 895 iterator iterator_to(reference value); 896 897 //! @copydoc ::boost::intrusive::rbtree::iterator_to(const_reference)const 898 const_iterator iterator_to(const_reference value) const; 899 900 //! @copydoc ::boost::intrusive::rbtree::init_node(reference) 901 static void init_node(reference value); 902 903 //! @copydoc ::boost::intrusive::rbtree::unlink_leftmost_without_rebalance 904 pointer unlink_leftmost_without_rebalance(); 905 906 //! @copydoc ::boost::intrusive::rbtree::replace_node 907 void replace_node(iterator replace_this, reference with_this); 908 909 //! @copydoc ::boost::intrusive::rbtree::remove_node 910 void remove_node(reference value); 911 912 //! @copydoc ::boost::intrusive::rbtree::merge_equal 913 template<class ...Options2> 914 void merge(multiset<T, Options2...> &source); 915 916 //! @copydoc ::boost::intrusive::rbtree::merge_equal 917 template<class ...Options2> 918 void merge(set<T, Options2...> &source); 919 920 #else 921 922 template<class Compare2> merge(multiset_impl<ValueTraits,VoidOrKeyOfValue,Compare2,SizeType,ConstantTimeSize,HeaderHolder> & source)923 void merge(multiset_impl<ValueTraits, VoidOrKeyOfValue, Compare2, SizeType, ConstantTimeSize, HeaderHolder> &source) 924 { return tree_type::merge_equal(source); } 925 926 template<class Compare2> merge(set_impl<ValueTraits,VoidOrKeyOfValue,Compare2,SizeType,ConstantTimeSize,HeaderHolder> & source)927 void merge(set_impl<ValueTraits, VoidOrKeyOfValue, Compare2, SizeType, ConstantTimeSize, HeaderHolder> &source) 928 { return tree_type::merge_equal(source); } 929 930 #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED 931 }; 932 933 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 934 935 template<class T, class ...Options> 936 bool operator!= (const multiset_impl<T, Options...> &x, const multiset_impl<T, Options...> &y); 937 938 template<class T, class ...Options> 939 bool operator>(const multiset_impl<T, Options...> &x, const multiset_impl<T, Options...> &y); 940 941 template<class T, class ...Options> 942 bool operator<=(const multiset_impl<T, Options...> &x, const multiset_impl<T, Options...> &y); 943 944 template<class T, class ...Options> 945 bool operator>=(const multiset_impl<T, Options...> &x, const multiset_impl<T, Options...> &y); 946 947 template<class T, class ...Options> 948 void swap(multiset_impl<T, Options...> &x, multiset_impl<T, Options...> &y); 949 950 #endif //#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) 951 952 //! Helper metafunction to define a \c multiset that yields to the same type when the 953 //! same options (either explicitly or implicitly) are used. 954 #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 955 template<class T, class ...Options> 956 #else 957 template<class T, class O1 = void, class O2 = void 958 , class O3 = void, class O4 = void 959 , class O5 = void, class O6 = void> 960 #endif 961 struct make_multiset 962 { 963 /// @cond 964 typedef typename pack_options 965 < rbtree_defaults, 966 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 967 O1, O2, O3, O4, O5, O6 968 #else 969 Options... 970 #endif 971 >::type packed_options; 972 973 typedef typename detail::get_value_traits 974 <T, typename packed_options::proto_value_traits>::type value_traits; 975 976 typedef multiset_impl 977 < value_traits 978 , typename packed_options::key_of_value 979 , typename packed_options::compare 980 , typename packed_options::size_type 981 , packed_options::constant_time_size 982 , typename packed_options::header_holder_type 983 > implementation_defined; 984 /// @endcond 985 typedef implementation_defined type; 986 }; 987 988 #ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED 989 990 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 991 template<class T, class O1, class O2, class O3, class O4, class O5, class O6> 992 #else 993 template<class T, class ...Options> 994 #endif 995 class multiset 996 : public make_multiset<T, 997 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 998 O1, O2, O3, O4, O5, O6 999 #else 1000 Options... 1001 #endif 1002 >::type 1003 { 1004 typedef typename make_multiset<T, 1005 #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES) 1006 O1, O2, O3, O4, O5, O6 1007 #else 1008 Options... 1009 #endif 1010 >::type Base; 1011 1012 BOOST_MOVABLE_BUT_NOT_COPYABLE(multiset) 1013 1014 public: 1015 typedef typename Base::key_compare key_compare; 1016 typedef typename Base::value_traits value_traits; 1017 typedef typename Base::iterator iterator; 1018 typedef typename Base::const_iterator const_iterator; 1019 1020 //Assert if passed value traits are compatible with the type 1021 BOOST_STATIC_ASSERT((detail::is_same<typename value_traits::value_type, T>::value)); 1022 multiset()1023 BOOST_INTRUSIVE_FORCEINLINE multiset() 1024 : Base() 1025 {} 1026 multiset(const key_compare & cmp,const value_traits & v_traits=value_traits ())1027 BOOST_INTRUSIVE_FORCEINLINE explicit multiset( const key_compare &cmp, const value_traits &v_traits = value_traits()) 1028 : Base(cmp, v_traits) 1029 {} 1030 1031 template<class Iterator> multiset(Iterator b,Iterator e,const key_compare & cmp=key_compare (),const value_traits & v_traits=value_traits ())1032 BOOST_INTRUSIVE_FORCEINLINE multiset( Iterator b, Iterator e 1033 , const key_compare &cmp = key_compare() 1034 , const value_traits &v_traits = value_traits()) 1035 : Base(b, e, cmp, v_traits) 1036 {} 1037 multiset(BOOST_RV_REF (multiset)x)1038 BOOST_INTRUSIVE_FORCEINLINE multiset(BOOST_RV_REF(multiset) x) 1039 : Base(BOOST_MOVE_BASE(Base, x)) 1040 {} 1041 operator =(BOOST_RV_REF (multiset)x)1042 BOOST_INTRUSIVE_FORCEINLINE multiset& operator=(BOOST_RV_REF(multiset) x) 1043 { return static_cast<multiset &>(this->Base::operator=(BOOST_MOVE_BASE(Base, x))); } 1044 1045 template <class Cloner, class Disposer> clone_from(const multiset & src,Cloner cloner,Disposer disposer)1046 BOOST_INTRUSIVE_FORCEINLINE void clone_from(const multiset &src, Cloner cloner, Disposer disposer) 1047 { Base::clone_from(src, cloner, disposer); } 1048 1049 template <class Cloner, class Disposer> clone_from(BOOST_RV_REF (multiset)src,Cloner cloner,Disposer disposer)1050 BOOST_INTRUSIVE_FORCEINLINE void clone_from(BOOST_RV_REF(multiset) src, Cloner cloner, Disposer disposer) 1051 { Base::clone_from(BOOST_MOVE_BASE(Base, src), cloner, disposer); } 1052 container_from_end_iterator(iterator end_iterator)1053 BOOST_INTRUSIVE_FORCEINLINE static multiset &container_from_end_iterator(iterator end_iterator) 1054 { return static_cast<multiset &>(Base::container_from_end_iterator(end_iterator)); } 1055 container_from_end_iterator(const_iterator end_iterator)1056 BOOST_INTRUSIVE_FORCEINLINE static const multiset &container_from_end_iterator(const_iterator end_iterator) 1057 { return static_cast<const multiset &>(Base::container_from_end_iterator(end_iterator)); } 1058 container_from_iterator(iterator it)1059 BOOST_INTRUSIVE_FORCEINLINE static multiset &container_from_iterator(iterator it) 1060 { return static_cast<multiset &>(Base::container_from_iterator(it)); } 1061 container_from_iterator(const_iterator it)1062 BOOST_INTRUSIVE_FORCEINLINE static const multiset &container_from_iterator(const_iterator it) 1063 { return static_cast<const multiset &>(Base::container_from_iterator(it)); } 1064 }; 1065 1066 #endif 1067 1068 } //namespace intrusive 1069 } //namespace boost 1070 1071 #include <boost/intrusive/detail/config_end.hpp> 1072 1073 #endif //BOOST_INTRUSIVE_SET_HPP 1074