| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160 | ///////////////////////////////////////////////////////////////////////////////// (C) Copyright Olaf Krzikalla 2004-2006.// (C) Copyright Ion Gaztanaga  2006-2013//// 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)//// See http://www.boost.org/libs/intrusive for documentation.///////////////////////////////////////////////////////////////////////////////#ifndef BOOST_INTRUSIVE_UNORDERED_SET_HPP#define BOOST_INTRUSIVE_UNORDERED_SET_HPP#include <boost/intrusive/detail/config_begin.hpp>#include <boost/intrusive/intrusive_fwd.hpp>#include <boost/intrusive/hashtable.hpp>#include <boost/move/move.hpp>#include <iterator>namespace boost {namespace intrusive {//! The class template unordered_set is an intrusive container, that mimics most of//! the interface of std::tr1::unordered_set as described in the C++ TR1.//!//! unordered_set is a semi-intrusive container: each object to be stored in the//! container must contain a proper hook, but the container also needs//! additional auxiliary memory to work: unordered_set needs a pointer to an array//! of type `bucket_type` to be passed in the constructor. This bucket array must//! have at least the same lifetime as the container. This makes the use of//! unordered_set more complicated than purely intrusive containers.//! `bucket_type` is default-constructible, copyable and assignable//!//! The template parameter \c T is the type to be managed by the container.//! The user can specify additional options and if no options are provided//! default options are used.//!//! The container supports the following options://! \c base_hook<>/member_hook<>/value_traits<>,//! \c constant_time_size<>, \c size_type<>, \c hash<> and \c equal<>//! \c bucket_traits<>, \c power_2_buckets<> and \c cache_begin<>.//!//! unordered_set only provides forward iterators but it provides 4 iterator types://! iterator and const_iterator to navigate through the whole container and//! local_iterator and const_local_iterator to navigate through the values//! stored in a single bucket. Local iterators are faster and smaller.//!//! It's not recommended to use non constant-time size unordered_sets because several//! key functions, like "empty()", become non-constant time functions. Non//! constant-time size unordered_sets are mainly provided to support auto-unlink hooks.//!//! unordered_set, unlike std::unordered_set, does not make automatic rehashings nor//! offers functions related to a load factor. Rehashing can be explicitly requested//! and the user must provide a new bucket array that will be used from that moment.//!//! Since no automatic rehashing is done, iterators are never invalidated when//! inserting or erasing elements. Iterators are only invalidated when rehasing.#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)template<class T, class ...Options>#elsetemplate<class ValueTraits, class Hash, class Equal, class SizeType, class BucketTraits, std::size_t BoolFlags>#endifclass unordered_set_impl   : public hashtable_impl<ValueTraits, Hash, Equal, SizeType, BucketTraits, BoolFlags>{   /// @cond   private:   typedef hashtable_impl<ValueTraits, Hash, Equal, SizeType, BucketTraits, BoolFlags> table_type;   //! This class is   //! movable   BOOST_MOVABLE_BUT_NOT_COPYABLE(unordered_set_impl)   typedef table_type implementation_defined;   /// @endcond   public:   typedef typename implementation_defined::value_type                  value_type;   typedef typename implementation_defined::value_traits                value_traits;   typedef typename implementation_defined::bucket_traits               bucket_traits;   typedef typename implementation_defined::pointer                     pointer;   typedef typename implementation_defined::const_pointer               const_pointer;   typedef typename implementation_defined::reference                   reference;   typedef typename implementation_defined::const_reference             const_reference;   typedef typename implementation_defined::difference_type             difference_type;   typedef typename implementation_defined::size_type                   size_type;   typedef typename implementation_defined::key_type                    key_type;   typedef typename implementation_defined::key_equal                   key_equal;   typedef typename implementation_defined::hasher                      hasher;   typedef typename implementation_defined::bucket_type                 bucket_type;   typedef typename implementation_defined::bucket_ptr                  bucket_ptr;   typedef typename implementation_defined::iterator                    iterator;   typedef typename implementation_defined::const_iterator              const_iterator;   typedef typename implementation_defined::insert_commit_data          insert_commit_data;   typedef typename implementation_defined::local_iterator              local_iterator;   typedef typename implementation_defined::const_local_iterator        const_local_iterator;   typedef typename implementation_defined::node_traits                 node_traits;   typedef typename implementation_defined::node                        node;   typedef typename implementation_defined::node_ptr                    node_ptr;   typedef typename implementation_defined::const_node_ptr              const_node_ptr;   typedef typename implementation_defined::node_algorithms             node_algorithms;   public:   //! <b>Requires</b>: buckets must not be being used by any other resource.   //!   //! <b>Effects</b>: Constructs an empty unordered_set_impl, storing a reference   //!   to the bucket array and copies of the hasher and equal functors.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If value_traits::node_traits::node   //!   constructor throws (this does not happen with predefined Boost.Intrusive hooks)   //!   or the copy constructor or invocation of Hash or Equal throws.   //!   //! <b>Notes</b>: buckets array must be disposed only after   //!   *this is disposed.   explicit unordered_set_impl( const bucket_traits &b_traits                              , const hasher & hash_func = hasher()                              , const key_equal &equal_func = key_equal()                              , const value_traits &v_traits = value_traits())      :  table_type(b_traits, hash_func, equal_func, v_traits)   {}   //! <b>Requires</b>: buckets must not be being used by any other resource   //!   and Dereferencing iterator must yield an lvalue of type value_type.   //!   //! <b>Effects</b>: Constructs an empty unordered_set and inserts elements from   //!   [b, e).   //!   //! <b>Complexity</b>: If N is std::distance(b, e): Average case is O(N)   //!   (with a good hash function and with buckets_len >= N),worst case O(N2).   //!   //! <b>Throws</b>: If value_traits::node_traits::node   //!   constructor throws (this does not happen with predefined Boost.Intrusive hooks)   //!   or the copy constructor or invocation of hasher or key_equal throws.   //!   //! <b>Notes</b>: buckets array must be disposed only after   //!   *this is disposed.   template<class Iterator>   unordered_set_impl( Iterator b                     , Iterator e                     , const bucket_traits &b_traits                     , const hasher & hash_func = hasher()                     , const key_equal &equal_func = key_equal()                     , const value_traits &v_traits = value_traits())      :  table_type(b_traits, hash_func, equal_func, v_traits)   {  table_type::insert_unique(b, e);  }   //! <b>Effects</b>: to-do   //!   unordered_set_impl(BOOST_RV_REF(unordered_set_impl) x)      :  table_type(::boost::move(static_cast<table_type&>(x)))   {}   //! <b>Effects</b>: to-do   //!   unordered_set_impl& operator=(BOOST_RV_REF(unordered_set_impl) x)   {  return static_cast<unordered_set_impl&>(table_type::operator=(::boost::move(static_cast<table_type&>(x)))); }   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Effects</b>: Detaches all elements from this. The objects in the unordered_set   //!   are not deleted (i.e. no destructors are called).   //!   //! <b>Complexity</b>: Linear to the number of elements in the unordered_set, if   //!   it's a safe-mode or auto-unlink value. Otherwise constant.   //!   //! <b>Throws</b>: Nothing.   ~unordered_set_impl()   {}   //! <b>Effects</b>: Returns an iterator pointing to the beginning of the unordered_set.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   iterator begin()   { return table_type::begin();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the beginning   //!   of the unordered_set.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   const_iterator begin() const   { return table_type::begin();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the beginning   //!   of the unordered_set.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   const_iterator cbegin() const   { return table_type::cbegin();  }   //! <b>Effects</b>: Returns an iterator pointing to the end of the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   iterator end()   { return table_type::end();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the end of the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_iterator end() const   { return table_type::end();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the end of the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_iterator cend() const   { return table_type::cend();  }   //! <b>Effects</b>: Returns the hasher object used by the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If hasher copy-constructor throws.   hasher hash_function() const   { return table_type::hash_function(); }   //! <b>Effects</b>: Returns the key_equal object used by the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If key_equal copy-constructor throws.   key_equal key_eq() const   { return table_type::key_eq(); }   //! <b>Effects</b>: Returns true if the container is empty.   //!   //! <b>Complexity</b>: if constant-time size and cache_last options are disabled,   //!   average constant time (worst case, with empty() == true: O(this->bucket_count()).   //!   Otherwise constant.   //!   //! <b>Throws</b>: Nothing.   bool empty() const   { return table_type::empty(); }   //! <b>Effects</b>: Returns the number of elements stored in the unordered_set.   //!   //! <b>Complexity</b>: Linear to elements contained in *this if   //!   constant-time size option is disabled. Constant-time otherwise.   //!   //! <b>Throws</b>: Nothing.   size_type size() const   { return table_type::size(); }   //! <b>Requires</b>: the hasher and the equality function unqualified swap   //!   call should not throw.   //!   //! <b>Effects</b>: Swaps the contents of two unordered_sets.   //!   Swaps also the contained bucket array and equality and hasher functors.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the swap() call for the comparison or hash functors   //!   found using ADL throw. Basic guarantee.   void swap(unordered_set_impl& other)   { table_type::swap(other.table_); }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   Cloner should yield to nodes that compare equal and produce the same   //!   hash than the original node.   //!   //! <b>Effects</b>: Erases all the elements from *this   //!   calling Disposer::operator()(pointer), clones all the   //!   elements from src calling Cloner::operator()(const_reference )   //!   and inserts them on *this. The hash function and the equality   //!   predicate are copied from the source.   //!   //!   If store_hash option is true, this method does not use the hash function.   //!   //!   If any operation throws, all cloned elements are unlinked and disposed   //!   calling Disposer::operator()(pointer).   //!   //! <b>Complexity</b>: Linear to erased plus inserted elements.   //!   //! <b>Throws</b>: If cloner or hasher throw or hash or equality predicate copying   //!   throws. Basic guarantee.   template <class Cloner, class Disposer>   void clone_from(const unordered_set_impl &src, Cloner cloner, Disposer disposer)   {  table_type::clone_from(src.table_, cloner, disposer);  }   #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Requires</b>: value must be an lvalue   //!   //! <b>Effects</b>: Tries to inserts value into the unordered_set.   //!   //! <b>Returns</b>: If the value   //!   is not already present inserts it and returns a pair containing the   //!   iterator to the new value and true. If there is an equivalent value   //!   returns a pair containing an iterator to the already present value   //!   and false.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Strong guarantee.   //!   //! <b>Note</b>: Does not affect the validity of iterators and references.   //!   No copy-constructors are called.   std::pair<iterator, bool> insert(reference value)   {  return table_type::insert_unique(value);  }   //! <b>Requires</b>: Dereferencing iterator must yield an lvalue   //!   of type value_type.   //!   //! <b>Effects</b>: Equivalent to this->insert(t) for each element in [b, e).   //!   //! <b>Complexity</b>: Average case O(N), where N is std::distance(b, e).   //!   Worst case O(N*this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Basic guarantee.   //!   //! <b>Note</b>: Does not affect the validity of iterators and references.   //!   No copy-constructors are called.   template<class Iterator>   void insert(Iterator b, Iterator e)   {  table_type::insert_unique(b, e);  }   //! <b>Requires</b>: "hasher" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hasher" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Checks if a value can be inserted in the unordered_set, using   //!   a user provided key instead of the value itself.   //!   //! <b>Returns</b>: If there is an equivalent value   //!   returns a pair containing an iterator to the already present value   //!   and false. If the value can be inserted returns true in the returned   //!   pair boolean and fills "commit_data" that is meant to be used with   //!   the "insert_commit" function.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If hasher or key_value_equal throw. Strong guarantee.   //!   //! <b>Notes</b>: This function is used to improve performance when constructing   //!   a value_type is expensive: if there is an equivalent value   //!   the constructed object must be discarded. Many times, the part of the   //!   node that is used to impose the hash or the equality is much cheaper to   //!   construct than the value_type and this function offers the possibility to   //!   use that the part to check if the insertion will be successful.   //!   //!   If the check is successful, the user can construct the value_type and use   //!   "insert_commit" to insert the object in constant-time.   //!   //!   "commit_data" remains valid for a subsequent "insert_commit" only if no more   //!   objects are inserted or erased from the unordered_set.   //!   //!   After a successful rehashing insert_commit_data remains valid.   template<class KeyType, class KeyHasher, class KeyValueEqual>   std::pair<iterator, bool> insert_check      (const KeyType &key, KeyHasher hasher, KeyValueEqual key_value_equal, insert_commit_data &commit_data)   {  return table_type::insert_unique_check(key, hasher, key_value_equal, commit_data); }   //! <b>Requires</b>: value must be an lvalue of type value_type. commit_data   //!   must have been obtained from a previous call to "insert_check".   //!   No objects should have been inserted or erased from the unordered_set between   //!   the "insert_check" that filled "commit_data" and the call to "insert_commit".   //!   //! <b>Effects</b>: Inserts the value in the unordered_set using the information obtained   //!   from the "commit_data" that a previous "insert_check" filled.   //!   //! <b>Returns</b>: An iterator to the newly inserted object.   //!   //! <b>Complexity</b>: Constant time.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Notes</b>: This function has only sense if a "insert_check" has been   //!   previously executed to fill "commit_data". No value should be inserted or   //!   erased between the "insert_check" and "insert_commit" calls.   //!   //!   After a successful rehashing insert_commit_data remains valid.   iterator insert_commit(reference value, const insert_commit_data &commit_data)   {  return table_type::insert_unique_commit(value, commit_data); }   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Effects</b>: Erases the element pointed to by i.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased element. No destructors are called.   void erase(const_iterator i)   {  table_type::erase(i);  }   //! <b>Effects</b>: Erases the range pointed to by b end e.   //!   //! <b>Complexity</b>: Average case O(std::distance(b, e)),   //!   worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   void erase(const_iterator b, const_iterator e)   {  table_type::erase(b, e);  }   //! <b>Effects</b>: Erases all the elements with the given value.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.  Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   size_type erase(const_reference value)   {  return table_type::erase(value);  }   //! <b>Requires</b>: "hasher" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hasher" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Erases all the elements that have the same hash and   //!   compare equal with the given key.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class KeyType, class KeyHasher, class KeyValueEqual>   size_type erase(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::erase(key, hash_func, equal_func);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases the element pointed to by i.   //!   Disposer::operator()(pointer) is called for the removed element.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class Disposer>   void erase_and_dispose(const_iterator i, Disposer disposer                              /// @cond                              , typename detail::enable_if_c<!detail::is_convertible<Disposer, const_iterator>::value >::type * = 0                              /// @endcond                              )   {  table_type::erase_and_dispose(i, disposer);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases the range pointed to by b end e.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Complexity</b>: Average case O(std::distance(b, e)),   //!   worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class Disposer>   void erase_and_dispose(const_iterator b, const_iterator e, Disposer disposer)   {  table_type::erase_and_dispose(b, e, disposer);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all the elements with the given value.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class Disposer>   size_type erase_and_dispose(const_reference value, Disposer disposer)   {  return table_type::erase_and_dispose(value, disposer);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all the elements with the given key.   //!   according to the comparison functor "equal_func".   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class KeyType, class KeyHasher, class KeyValueEqual, class Disposer>   size_type erase_and_dispose(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func, Disposer disposer)   {  return table_type::erase_and_dispose(key, hash_func, equal_func, disposer);  }   //! <b>Effects</b>: Erases all of the elements.   //!   //! <b>Complexity</b>: Linear to the number of elements on the container.   //!   if it's a safe-mode or auto-unlink value_type. Constant time otherwise.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   void clear()   {  return table_type::clear();  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all of the elements.   //!   //! <b>Complexity</b>: Linear to the number of elements on the container.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class Disposer>   void clear_and_dispose(Disposer disposer)   {  return table_type::clear_and_dispose(disposer);  }   //! <b>Effects</b>: Returns the number of contained elements with the given value   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   size_type count(const_reference value) const   {  return table_type::find(value) != end();  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "equal_func" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "equal_func" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns the number of contained elements with the given key   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw.   template<class KeyType, class KeyHasher, class KeyValueEqual>   size_type count(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::find(key, hash_func, equal_func) != end();  }   //! <b>Effects</b>: Finds an iterator to the first element is equal to   //!   "value" or end() if that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   iterator find(const_reference value)   {  return table_type::find(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "equal_func" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "equal_func" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Finds an iterator to the first element whose key is   //!   "key" according to the given hasher and equality functor or end() if   //!   that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   iterator find(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::find(key, hash_func, equal_func);  }   //! <b>Effects</b>: Finds a const_iterator to the first element whose key is   //!   "key" or end() if that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   const_iterator find(const_reference value) const   {  return table_type::find(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "equal_func" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "equal_func" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Finds an iterator to the first element whose key is   //!   "key" according to the given hasher and equality functor or end() if   //!   that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   const_iterator find(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::find(key, hash_func, equal_func);  }   //! <b>Effects</b>: Returns a range containing all elements with values equivalent   //!   to value. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(value)). Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   std::pair<iterator,iterator> equal_range(const_reference value)   {  return table_type::equal_range(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "equal_func" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "equal_func" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns a range containing all elements with equivalent   //!   keys. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(key, hash_func, hash_func)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or the equal_func throw.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   std::pair<iterator,iterator> equal_range(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::equal_range(key, hash_func, equal_func);  }   //! <b>Effects</b>: Returns a range containing all elements with values equivalent   //!   to value. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(value)). Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   std::pair<const_iterator, const_iterator>      equal_range(const_reference value) const   {  return table_type::equal_range(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "equal_func" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "equal_func" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns a range containing all elements with equivalent   //!   keys. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(key, hash_func, equal_func)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the hash_func or equal_func throw.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   std::pair<const_iterator, const_iterator>      equal_range(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::equal_range(key, hash_func, equal_func);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid iterator belonging to the unordered_set   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the internal hash function throws.   iterator iterator_to(reference value)   {  return table_type::iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_iterator belonging to the   //!   unordered_set that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the internal hash function throws.   const_iterator iterator_to(const_reference value) const   {  return table_type::iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid local_iterator belonging to the unordered_set   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: This static function is available only if the <i>value traits</i>   //!   is stateless.   static local_iterator s_local_iterator_to(reference value)   {  return table_type::s_local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_local_iterator belonging to   //!   the unordered_set that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: This static function is available only if the <i>value traits</i>   //!   is stateless.   static const_local_iterator s_local_iterator_to(const_reference value)   {  return table_type::s_local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid local_iterator belonging to the unordered_set   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   local_iterator local_iterator_to(reference value)   {  return table_type::local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_local_iterator belonging to   //!   the unordered_set that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_local_iterator local_iterator_to(const_reference value) const   {  return table_type::local_iterator_to(value);  }   //! <b>Effects</b>: Returns the number of buckets passed in the constructor   //!   or the last rehash function.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   size_type bucket_count() const   {  return table_type::bucket_count();   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns the number of elements in the nth bucket.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   size_type bucket_size(size_type n) const   {  return table_type::bucket_size(n);   }   //! <b>Effects</b>: Returns the index of the bucket in which elements   //!   with keys equivalent to k would be found, if any such element existed.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the hash functor throws.   //!   //! <b>Note</b>: the return value is in the range [0, this->bucket_count()).   size_type bucket(const value_type& k) const   {  return table_type::bucket(k);   }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //! <b>Effects</b>: Returns the index of the bucket in which elements   //!   with keys equivalent to k would be found, if any such element existed.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If hash_func throws.   //!   //! <b>Note</b>: the return value is in the range [0, this->bucket_count()).   template<class KeyType, class KeyHasher>   size_type bucket(const KeyType& k,  KeyHasher hash_func) const   {  return table_type::bucket(k, hash_func);   }   //! <b>Effects</b>: Returns the bucket array pointer passed in the constructor   //!   or the last rehash function.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   bucket_ptr bucket_pointer() const   {  return table_type::bucket_pointer();   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   local_iterator begin(size_type n)   {  return table_type::begin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator begin(size_type n) const   {  return table_type::begin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator cbegin(size_type n) const   {  return table_type::cbegin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   local_iterator end(size_type n)   {  return table_type::end(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator end(size_type n) const   {  return table_type::end(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator cend(size_type n) const   {  return table_type::cend(n);   }   //! <b>Requires</b>: new_buckets must be a pointer to a new bucket array   //!   or the same as the old bucket array. new_size is the length of the   //!   the array pointed by new_buckets. If new_buckets == this->bucket_pointer()   //!   n can be bigger or smaller than this->bucket_count().   //!   //! <b>Effects</b>: Updates the internal reference with the new bucket erases   //!   the values from the old bucket and inserts then in the new one.   //!   //!   If store_hash option is true, this method does not use the hash function.   //!   //! <b>Complexity</b>: Average case linear in this->size(), worst case quadratic.   //!   //! <b>Throws</b>: If the hasher functor throws. Basic guarantee.   void rehash(const bucket_traits &new_bucket_traits)   {  table_type::rehash(new_bucket_traits); }   //! <b>Requires</b>:   //!   //! <b>Effects</b>:   //!   //! <b>Complexity</b>:   //!   //! <b>Throws</b>:   //!   //! <b>Note</b>: this method is only available if incremental<true> option is activated.   bool incremental_rehash(bool grow = true)   {  return table_type::incremental_rehash(grow);  }   //! <b>Note</b>: this method is only available if incremental<true> option is activated.   bool incremental_rehash(const bucket_traits &new_bucket_traits)   {  return table_type::incremental_rehash(new_bucket_traits);  }   //! <b>Requires</b>:   //!   //! <b>Effects</b>:   //!   //! <b>Complexity</b>:   //!   //! <b>Throws</b>:   size_type split_count() const   {  return table_type::split_count(); }   //! <b>Effects</b>: Returns the nearest new bucket count optimized for   //!   the container that is bigger than n. This suggestion can be used   //!   to create bucket arrays with a size that will usually improve   //!   container's performance. If such value does not exist, the   //!   higher possible value is returned.   //!   //! <b>Complexity</b>: Amortized constant time.   //!   //! <b>Throws</b>: Nothing.   static size_type suggested_upper_bucket_count(size_type n)   {  return table_type::suggested_upper_bucket_count(n);  }   //! <b>Effects</b>: Returns the nearest new bucket count optimized for   //!   the container that is smaller than n. This suggestion can be used   //!   to create bucket arrays with a size that will usually improve   //!   container's performance. If such value does not exist, the   //!   lower possible value is returned.   //!   //! <b>Complexity</b>: Amortized constant time.   //!   //! <b>Throws</b>: Nothing.   static size_type suggested_lower_bucket_count(size_type n)   {  return table_type::suggested_lower_bucket_count(n);  }   #endif   //   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED};//! Helper metafunction to define an \c unordered_set that yields to the same type when the//! same options (either explicitly or implicitly) are used.#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)template<class T, class ...Options>#elsetemplate<class T, class O1 = void, class O2 = void                , class O3 = void, class O4 = void                , class O5 = void, class O6 = void                , class O7 = void, class O8 = void                , class O9 = void, class O10= void                >#endifstruct make_unordered_set{   /// @cond   typedef typename pack_options      < hashtable_defaults,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type packed_options;   typedef typename detail::get_value_traits      <T, typename packed_options::proto_value_traits>::type value_traits;   typedef typename make_real_bucket_traits            <T, true, packed_options>::type real_bucket_traits;   typedef unordered_set_impl      < value_traits      , typename packed_options::hash      , typename packed_options::equal      , typename packed_options::size_type      , real_bucket_traits      ,  (std::size_t(true)*hash_bool_flags::unique_keys_pos)      |  (std::size_t(packed_options::constant_time_size)*hash_bool_flags::constant_time_size_pos)      |  (std::size_t(packed_options::power_2_buckets)*hash_bool_flags::power_2_buckets_pos)      |  (std::size_t(packed_options::cache_begin)*hash_bool_flags::cache_begin_pos)      |  (std::size_t(packed_options::compare_hash)*hash_bool_flags::compare_hash_pos)      |  (std::size_t(packed_options::incremental)*hash_bool_flags::incremental_pos)      > implementation_defined;   /// @endcond   typedef implementation_defined type;};#ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED#if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)template<class T, class O1, class O2, class O3, class O4, class O5, class O6, class O7, class O8, class O9, class O10>#elsetemplate<class T, class ...Options>#endifclass unordered_set   :  public make_unordered_set<T,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type{   typedef typename make_unordered_set      <T,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type   Base;   //Assert if passed value traits are compatible with the type   BOOST_STATIC_ASSERT((detail::is_same<typename Base::value_traits::value_type, T>::value));   BOOST_MOVABLE_BUT_NOT_COPYABLE(unordered_set)   public:   typedef typename Base::value_traits       value_traits;   typedef typename Base::bucket_traits      bucket_traits;   typedef typename Base::iterator           iterator;   typedef typename Base::const_iterator     const_iterator;   typedef typename Base::bucket_ptr         bucket_ptr;   typedef typename Base::size_type          size_type;   typedef typename Base::hasher             hasher;   typedef typename Base::key_equal          key_equal;   explicit unordered_set  ( const bucket_traits &b_traits                           , const hasher & hash_func = hasher()                           , const key_equal &equal_func = key_equal()                           , const value_traits &v_traits = value_traits())      :  Base(b_traits, hash_func, equal_func, v_traits)   {}   template<class Iterator>   unordered_set  ( Iterator b                  , Iterator e                  , const bucket_traits &b_traits                  , const hasher & hash_func = hasher()                  , const key_equal &equal_func = key_equal()                  , const value_traits &v_traits = value_traits())      :  Base(b, e, b_traits, hash_func, equal_func, v_traits)   {}   unordered_set(BOOST_RV_REF(unordered_set) x)      :  Base(::boost::move(static_cast<Base&>(x)))   {}   unordered_set& operator=(BOOST_RV_REF(unordered_set) x)   {  return static_cast<unordered_set&>(this->Base::operator=(::boost::move(static_cast<Base&>(x))));  }};#endif//! The class template unordered_multiset is an intrusive container, that mimics most of//! the interface of std::tr1::unordered_multiset as described in the C++ TR1.//!//! unordered_multiset is a semi-intrusive container: each object to be stored in the//! container must contain a proper hook, but the container also needs//! additional auxiliary memory to work: unordered_multiset needs a pointer to an array//! of type `bucket_type` to be passed in the constructor. This bucket array must//! have at least the same lifetime as the container. This makes the use of//! unordered_multiset more complicated than purely intrusive containers.//! `bucket_type` is default-constructible, copyable and assignable//!//! The template parameter \c T is the type to be managed by the container.//! The user can specify additional options and if no options are provided//! default options are used.//!//! The container supports the following options://! \c base_hook<>/member_hook<>/value_traits<>,//! \c constant_time_size<>, \c size_type<>, \c hash<> and \c equal<>//! \c bucket_traits<>, \c power_2_buckets<> and \c cache_begin<>.//!//! unordered_multiset only provides forward iterators but it provides 4 iterator types://! iterator and const_iterator to navigate through the whole container and//! local_iterator and const_local_iterator to navigate through the values//! stored in a single bucket. Local iterators are faster and smaller.//!//! It's not recommended to use non constant-time size unordered_multisets because several//! key functions, like "empty()", become non-constant time functions. Non//! constant-time size unordered_multisets are mainly provided to support auto-unlink hooks.//!//! unordered_multiset, unlike std::unordered_set, does not make automatic rehashings nor//! offers functions related to a load factor. Rehashing can be explicitly requested//! and the user must provide a new bucket array that will be used from that moment.//!//! Since no automatic rehashing is done, iterators are never invalidated when//! inserting or erasing elements. Iterators are only invalidated when rehasing.#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)template<class T, class ...Options>#elsetemplate<class ValueTraits, class Hash, class Equal, class SizeType, class BucketTraits, std::size_t BoolFlags>#endifclass unordered_multiset_impl   : public hashtable_impl<ValueTraits, Hash, Equal, SizeType, BucketTraits, BoolFlags>{   /// @cond   private:   typedef hashtable_impl<ValueTraits, Hash, Equal, SizeType, BucketTraits, BoolFlags> table_type;   /// @endcond   //Movable   BOOST_MOVABLE_BUT_NOT_COPYABLE(unordered_multiset_impl)   typedef table_type implementation_defined;   public:   typedef typename implementation_defined::value_type                  value_type;   typedef typename implementation_defined::value_traits                value_traits;   typedef typename implementation_defined::bucket_traits               bucket_traits;   typedef typename implementation_defined::pointer                     pointer;   typedef typename implementation_defined::const_pointer               const_pointer;   typedef typename implementation_defined::reference                   reference;   typedef typename implementation_defined::const_reference             const_reference;   typedef typename implementation_defined::difference_type             difference_type;   typedef typename implementation_defined::size_type                   size_type;   typedef typename implementation_defined::key_type                    key_type;   typedef typename implementation_defined::key_equal                   key_equal;   typedef typename implementation_defined::hasher                      hasher;   typedef typename implementation_defined::bucket_type                 bucket_type;   typedef typename implementation_defined::bucket_ptr                  bucket_ptr;   typedef typename implementation_defined::iterator                    iterator;   typedef typename implementation_defined::const_iterator              const_iterator;   typedef typename implementation_defined::insert_commit_data          insert_commit_data;   typedef typename implementation_defined::local_iterator              local_iterator;   typedef typename implementation_defined::const_local_iterator        const_local_iterator;   typedef typename implementation_defined::node_traits                 node_traits;   typedef typename implementation_defined::node                        node;   typedef typename implementation_defined::node_ptr                    node_ptr;   typedef typename implementation_defined::const_node_ptr              const_node_ptr;   typedef typename implementation_defined::node_algorithms             node_algorithms;   public:   //! <b>Requires</b>: buckets must not be being used by any other resource.   //!   //! <b>Effects</b>: Constructs an empty unordered_multiset, storing a reference   //!   to the bucket array and copies of the hasher and equal functors.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If value_traits::node_traits::node   //!   constructor throws (this does not happen with predefined Boost.Intrusive hooks)   //!   or the copy constructor or invocation of Hash or Equal throws.   //!   //! <b>Notes</b>: buckets array must be disposed only after   //!   *this is disposed.   explicit unordered_multiset_impl ( const bucket_traits &b_traits                                    , const hasher & hash_func = hasher()                                    , const key_equal &equal_func = key_equal()                                    , const value_traits &v_traits = value_traits())      :  table_type(b_traits, hash_func, equal_func, v_traits)   {}   //! <b>Requires</b>: buckets must not be being used by any other resource   //!   and Dereferencing iterator must yield an lvalue of type value_type.   //!   //! <b>Effects</b>: Constructs an empty unordered_multiset and inserts elements from   //!   [b, e).   //!   //! <b>Complexity</b>: If N is std::distance(b, e): Average case is O(N)   //!   (with a good hash function and with buckets_len >= N),worst case O(N2).   //!   //! <b>Throws</b>: If value_traits::node_traits::node   //!   constructor throws (this does not happen with predefined Boost.Intrusive hooks)   //!   or the copy constructor or invocation of hasher or key_equal throws.   //!   //! <b>Notes</b>: buckets array must be disposed only after   //!   *this is disposed.   template<class Iterator>   unordered_multiset_impl ( Iterator b                           , Iterator e                           , const bucket_traits &b_traits                           , const hasher & hash_func = hasher()                           , const key_equal &equal_func = key_equal()                           , const value_traits &v_traits = value_traits())      :  table_type(b_traits, hash_func, equal_func, v_traits)   {  table_type::insert_equal(b, e);  }   //! <b>Effects</b>: to-do   //!   unordered_multiset_impl(BOOST_RV_REF(unordered_multiset_impl) x)      :  table_type(::boost::move(static_cast<table_type&>(x)))   {}   //! <b>Effects</b>: to-do   //!   unordered_multiset_impl& operator=(BOOST_RV_REF(unordered_multiset_impl) x)   {  return static_cast<unordered_multiset_impl&>(table_type::operator=(::boost::move(static_cast<table_type&>(x))));  }   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Effects</b>: Detaches all elements from this. The objects in the unordered_multiset   //!   are not deleted (i.e. no destructors are called).   //!   //! <b>Complexity</b>: Linear to the number of elements in the unordered_multiset, if   //!   it's a safe-mode or auto-unlink value. Otherwise constant.   //!   //! <b>Throws</b>: Nothing.   ~unordered_multiset_impl()   {}   //! <b>Effects</b>: Returns an iterator pointing to the beginning of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   iterator begin()   { return table_type::begin();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the beginning   //!   of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   const_iterator begin() const   { return table_type::begin();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the beginning   //!   of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant time if `cache_begin<>` is true. Amortized   //!   constant time with worst case (empty unordered_set) O(this->bucket_count())   //!   //! <b>Throws</b>: Nothing.   const_iterator cbegin() const   { return table_type::cbegin();  }   //! <b>Effects</b>: Returns an iterator pointing to the end of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   iterator end()   { return table_type::end();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the end of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_iterator end() const   { return table_type::end();  }   //! <b>Effects</b>: Returns a const_iterator pointing to the end of the unordered_multiset.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_iterator cend() const   { return table_type::cend();  }   //! <b>Effects</b>: Returns the hasher object used by the unordered_set.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If hasher copy-constructor throws.   hasher hash_function() const   { return table_type::hash_function(); }   //! <b>Effects</b>: Returns the key_equal object used by the unordered_multiset.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If key_equal copy-constructor throws.   key_equal key_eq() const   { return table_type::key_eq(); }   //! <b>Effects</b>: Returns true if the container is empty.   //!   //! <b>Complexity</b>: if constant-time size and cache_last options are disabled,   //!   average constant time (worst case, with empty() == true: O(this->bucket_count()).   //!   Otherwise constant.   //!   //! <b>Throws</b>: Nothing.   bool empty() const   { return table_type::empty(); }   //! <b>Effects</b>: Returns the number of elements stored in the unordered_multiset.   //!   //! <b>Complexity</b>: Linear to elements contained in *this if   //!   constant-time size option is disabled. Constant-time otherwise.   //!   //! <b>Throws</b>: Nothing.   size_type size() const   { return table_type::size(); }   //! <b>Requires</b>: the hasher and the equality function unqualified swap   //!   call should not throw.   //!   //! <b>Effects</b>: Swaps the contents of two unordered_multisets.   //!   Swaps also the contained bucket array and equality and hasher functors.   //!   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the swap() call for the comparison or hash functors   //!   found using ADL throw. Basic guarantee.   void swap(unordered_multiset_impl& other)   { table_type::swap(other.table_); }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   Cloner should yield to nodes that compare equal and produce the same   //!   hash than the original node.   //!   //! <b>Effects</b>: Erases all the elements from *this   //!   calling Disposer::operator()(pointer), clones all the   //!   elements from src calling Cloner::operator()(const_reference )   //!   and inserts them on *this. The hash function and the equality   //!   predicate are copied from the source.   //!   //!   If store_hash option is true, this method does not use the hash function.   //!   //!   If any operation throws, all cloned elements are unlinked and disposed   //!   calling Disposer::operator()(pointer).   //!   //! <b>Complexity</b>: Linear to erased plus inserted elements.   //!   //! <b>Throws</b>: If cloner or hasher throw or hash or equality predicate copying   //!   throws. Basic guarantee.   template <class Cloner, class Disposer>   void clone_from(const unordered_multiset_impl &src, Cloner cloner, Disposer disposer)   {  table_type::clone_from(src.table_, cloner, disposer);  }   #endif   //   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Requires</b>: value must be an lvalue   //!   //! <b>Effects</b>: Inserts value into the unordered_multiset.   //!   //! <b>Returns</b>: An iterator to the new inserted value.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Strong guarantee.   //!   //! <b>Note</b>: Does not affect the validity of iterators and references.   //!   No copy-constructors are called.   iterator insert(reference value)   {  return table_type::insert_equal(value);  }   //! <b>Requires</b>: Dereferencing iterator must yield an lvalue   //!   of type value_type.   //!   //! <b>Effects</b>: Equivalent to this->insert(t) for each element in [b, e).   //!   //! <b>Complexity</b>: Average case is O(N), where N is the   //!   size of the range.   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Basic guarantee.   //!   //! <b>Note</b>: Does not affect the validity of iterators and references.   //!   No copy-constructors are called.   template<class Iterator>   void insert(Iterator b, Iterator e)   {  table_type::insert_equal(b, e);  }   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED   //! <b>Effects</b>: Erases the element pointed to by i.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased element. No destructors are called.   void erase(const_iterator i)   {  table_type::erase(i);  }   //! <b>Effects</b>: Erases the range pointed to by b end e.   //!   //! <b>Complexity</b>: Average case O(std::distance(b, e)),   //!   worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   void erase(const_iterator b, const_iterator e)   {  table_type::erase(b, e);  }   //! <b>Effects</b>: Erases all the elements with the given value.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   size_type erase(const_reference value)   {  return table_type::erase(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Erases all the elements that have the same hash and   //!   compare equal with the given key.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the hash_func or the equal_func functors throws.   //!   Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class KeyType, class KeyHasher, class KeyValueEqual>   size_type erase(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::erase(key, hash_func, equal_func);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases the element pointed to by i.   //!   Disposer::operator()(pointer) is called for the removed element.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class Disposer>   void erase_and_dispose(const_iterator i, Disposer disposer                              /// @cond                              , typename detail::enable_if_c<!detail::is_convertible<Disposer, const_iterator>::value >::type * = 0                              /// @endcond                              )   {  table_type::erase_and_dispose(i, disposer);  }   #if !defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)   template<class Disposer>   void erase_and_dispose(const_iterator i, Disposer disposer)   {  this->erase_and_dispose(const_iterator(i), disposer);   }   #endif   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases the range pointed to by b end e.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Complexity</b>: Average case O(std::distance(b, e)),   //!   worst case O(this->size()).   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class Disposer>   void erase_and_dispose(const_iterator b, const_iterator e, Disposer disposer)   {  table_type::erase_and_dispose(b, e, disposer);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all the elements with the given value.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class Disposer>   size_type erase_and_dispose(const_reference value, Disposer disposer)   {  return table_type::erase_and_dispose(value, disposer);  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all the elements with the given key.   //!   according to the comparison functor "equal_func".   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Returns</b>: The number of erased elements.   //!   //! <b>Complexity</b>: Average case O(this->count(value)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If hash_func or equal_func throw. Basic guarantee.   //!   //! <b>Note</b>: Invalidates the iterators   //!    to the erased elements.   template<class KeyType, class KeyHasher, class KeyValueEqual, class Disposer>   size_type erase_and_dispose(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func, Disposer disposer)   {  return table_type::erase_and_dispose(key, hash_func, equal_func, disposer);  }   //! <b>Effects</b>: Erases all the elements of the container.   //!   //! <b>Complexity</b>: Linear to the number of elements on the container.   //!   if it's a safe-mode or auto-unlink value_type. Constant time otherwise.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   void clear()   {  return table_type::clear();  }   //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.   //!   //! <b>Effects</b>: Erases all the elements of the container.   //!   //! <b>Complexity</b>: Linear to the number of elements on the container.   //!   Disposer::operator()(pointer) is called for the removed elements.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: Invalidates the iterators (but not the references)   //!    to the erased elements. No destructors are called.   template<class Disposer>   void clear_and_dispose(Disposer disposer)   {  return table_type::clear_and_dispose(disposer);  }   //! <b>Effects</b>: Returns the number of contained elements with the given key   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   size_type count(const_reference value) const   {  return table_type::count(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns the number of contained elements with the given key   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   template<class KeyType, class KeyHasher, class KeyValueEqual>   size_type count(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::count(key, hash_func, equal_func);  }   //! <b>Effects</b>: Finds an iterator to the first element whose value is   //!   "value" or end() if that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   iterator find(const_reference value)   {  return table_type::find(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Finds an iterator to the first element whose key is   //!   "key" according to the given hasher and equality functor or end() if   //!   that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   iterator find(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::find(key, hash_func, equal_func);  }   //! <b>Effects</b>: Finds a const_iterator to the first element whose key is   //!   "key" or end() if that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   const_iterator find(const_reference value) const   {  return table_type::find(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Finds an iterator to the first element whose key is   //!   "key" according to the given hasher and equality functor or end() if   //!   that element does not exist.   //!   //! <b>Complexity</b>: Average case O(1), worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   const_iterator find(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::find(key, hash_func, equal_func);  }   //! <b>Effects</b>: Returns a range containing all elements with values equivalent   //!   to value. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(value)). Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   std::pair<iterator,iterator> equal_range(const_reference value)   {  return table_type::equal_range(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns a range containing all elements with equivalent   //!   keys. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(key, hash_func, equal_func)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   std::pair<iterator,iterator> equal_range      (const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func)   {  return table_type::equal_range(key, hash_func, equal_func);  }   //! <b>Effects</b>: Returns a range containing all elements with values equivalent   //!   to value. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(value)). Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   std::pair<const_iterator, const_iterator>      equal_range(const_reference value) const   {  return table_type::equal_range(value);  }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //!   "key_value_equal" must be a equality function that induces   //!   the same equality as key_equal. The difference is that   //!   "key_value_equal" compares an arbitrary key with the contained values.   //!   //! <b>Effects</b>: Returns a range containing all elements with equivalent   //!   keys. Returns std::make_pair(this->end(), this->end()) if no such   //!   elements exist.   //!   //! <b>Complexity</b>: Average case O(this->count(key, hash_func, equal_func)).   //!   Worst case O(this->size()).   //!   //! <b>Throws</b>: If the internal hasher or the equality functor throws.   //!   //! <b>Note</b>: This function is used when constructing a value_type   //!   is expensive and the value_type can be compared with a cheaper   //!   key type. Usually this key is part of the value_type.   template<class KeyType, class KeyHasher, class KeyValueEqual>   std::pair<const_iterator, const_iterator>      equal_range(const KeyType& key, KeyHasher hash_func, KeyValueEqual equal_func) const   {  return table_type::equal_range(key, hash_func, equal_func);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_multiset of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid iterator belonging to the unordered_multiset   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the hash function throws.   iterator iterator_to(reference value)   {  return table_type::iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_multiset of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_iterator belonging to the   //!   unordered_multiset that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the hash function throws.   const_iterator iterator_to(const_reference value) const   {  return table_type::iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid local_iterator belonging to the unordered_set   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: This static function is available only if the <i>value traits</i>   //!   is stateless.   static local_iterator s_local_iterator_to(reference value)   {  return table_type::s_local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_local_iterator belonging to   //!   the unordered_set that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>: This static function is available only if the <i>value traits</i>   //!   is stateless.   static const_local_iterator s_local_iterator_to(const_reference value)   {  return table_type::s_local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid local_iterator belonging to the unordered_set   //!   that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   local_iterator local_iterator_to(reference value)   {  return table_type::local_iterator_to(value);  }   //! <b>Requires</b>: value must be an lvalue and shall be in a unordered_set of   //!   appropriate type. Otherwise the behavior is undefined.   //!   //! <b>Effects</b>: Returns: a valid const_local_iterator belonging to   //!   the unordered_set that points to the value   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   const_local_iterator local_iterator_to(const_reference value) const   {  return table_type::local_iterator_to(value);  }   //! <b>Effects</b>: Returns the number of buckets passed in the constructor   //!   or the last rehash function.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   size_type bucket_count() const   {  return table_type::bucket_count();   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns the number of elements in the nth bucket.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   size_type bucket_size(size_type n) const   {  return table_type::bucket_size(n);   }   //! <b>Effects</b>: Returns the index of the bucket in which elements   //!   with keys equivalent to k would be found, if any such element existed.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the hash functor throws.   //!   //! <b>Note</b>: the return value is in the range [0, this->bucket_count()).   size_type bucket(const value_type& k) const   {  return table_type::bucket(k);   }   //! <b>Requires</b>: "hash_func" must be a hash function that induces   //!   the same hash values as the stored hasher. The difference is that   //!   "hash_func" hashes the given key instead of the value_type.   //!   //! <b>Effects</b>: Returns the index of the bucket in which elements   //!   with keys equivalent to k would be found, if any such element existed.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: If the hash functor throws.   //!   //! <b>Note</b>: the return value is in the range [0, this->bucket_count()).   template<class KeyType, class KeyHasher>   size_type bucket(const KeyType& k, const KeyHasher &hash_func) const   {  return table_type::bucket(k, hash_func);   }   //! <b>Effects</b>: Returns the bucket array pointer passed in the constructor   //!   or the last rehash function.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   bucket_ptr bucket_pointer() const   {  return table_type::bucket_pointer();   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   local_iterator begin(size_type n)   {  return table_type::begin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator begin(size_type n) const   {  return table_type::begin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the beginning   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator cbegin(size_type n) const   {  return table_type::cbegin(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   local_iterator end(size_type n)   {  return table_type::end(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator end(size_type n) const   {  return table_type::end(n);   }   //! <b>Requires</b>: n is in the range [0, this->bucket_count()).   //!   //! <b>Effects</b>: Returns a const_local_iterator pointing to the end   //!   of the sequence stored in the bucket n.   //!   //! <b>Complexity</b>: Constant.   //!   //! <b>Throws</b>: Nothing.   //!   //! <b>Note</b>:  [this->begin(n), this->end(n)) is a valid range   //!   containing all of the elements in the nth bucket.   const_local_iterator cend(size_type n) const   {  return table_type::cend(n);   }   //! <b>Requires</b>: new_buckets must be a pointer to a new bucket array   //!   or the same as the old bucket array. new_size is the length of the   //!   the array pointed by new_buckets. If new_buckets == this->bucket_pointer()   //!   n can be bigger or smaller than this->bucket_count().   //!   //! <b>Effects</b>: Updates the internal reference with the new bucket erases   //!   the values from the old bucket and inserts then in the new one.   //!   //!   If store_hash option is true, this method does not use the hash function.   //!   //! <b>Complexity</b>: Average case linear in this->size(), worst case quadratic.   //!   //! <b>Throws</b>: If the hasher functor throws.   void rehash(const bucket_traits &new_bucket_traits)   {  table_type::rehash(new_bucket_traits); }   //! <b>Requires</b>:   //!   //! <b>Effects</b>:   //!   //! <b>Complexity</b>:   //!   //! <b>Throws</b>:   //!   //! <b>Note</b>: this method is only available if incremental<true> option is activated.   bool incremental_rehash(bool grow = true)   {  return table_type::incremental_rehash(grow);  }   //! <b>Note</b>: this method is only available if incremental<true> option is activated.   bool incremental_rehash(const bucket_traits &new_bucket_traits)   {  return table_type::incremental_rehash(new_bucket_traits);  }   //! <b>Requires</b>:   //!   //! <b>Effects</b>:   //!   //! <b>Complexity</b>:   //!   //! <b>Throws</b>:   size_type split_count() const   {  return table_type::split_count(); }   //! <b>Effects</b>: Returns the nearest new bucket count optimized for   //!   the container that is bigger than n. This suggestion can be used   //!   to create bucket arrays with a size that will usually improve   //!   container's performance. If such value does not exist, the   //!   higher possible value is returned.   //!   //! <b>Complexity</b>: Amortized constant time.   //!   //! <b>Throws</b>: Nothing.   static size_type suggested_upper_bucket_count(size_type n)   {  return table_type::suggested_upper_bucket_count(n);  }   //! <b>Effects</b>: Returns the nearest new bucket count optimized for   //!   the container that is smaller than n. This suggestion can be used   //!   to create bucket arrays with a size that will usually improve   //!   container's performance. If such value does not exist, the   //!   lower possible value is returned.   //!   //! <b>Complexity</b>: Amortized constant time.   //!   //! <b>Throws</b>: Nothing.   static size_type suggested_lower_bucket_count(size_type n)   {  return table_type::suggested_lower_bucket_count(n);  }   #endif   //   #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED};//! Helper metafunction to define an \c unordered_multiset that yields to the same type when the//! same options (either explicitly or implicitly) are used.#if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)template<class T, class ...Options>#elsetemplate<class T, class O1 = void, class O2 = void                , class O3 = void, class O4 = void                , class O5 = void, class O6 = void                , class O7 = void, class O8 = void                , class O9 = void, class O10= void                >#endifstruct make_unordered_multiset{   /// @cond   typedef typename pack_options      < hashtable_defaults,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type packed_options;   typedef typename detail::get_value_traits      <T, typename packed_options::proto_value_traits>::type value_traits;   typedef typename make_real_bucket_traits            <T, true, packed_options>::type real_bucket_traits;   typedef unordered_multiset_impl      < value_traits      , typename packed_options::hash      , typename packed_options::equal      , typename packed_options::size_type      , real_bucket_traits      ,  (std::size_t(false)*hash_bool_flags::unique_keys_pos)      |  (std::size_t(packed_options::constant_time_size)*hash_bool_flags::constant_time_size_pos)      |  (std::size_t(packed_options::power_2_buckets)*hash_bool_flags::power_2_buckets_pos)      |  (std::size_t(packed_options::cache_begin)*hash_bool_flags::cache_begin_pos)      |  (std::size_t(packed_options::compare_hash)*hash_bool_flags::compare_hash_pos)      |  (std::size_t(packed_options::incremental)*hash_bool_flags::incremental_pos)      > implementation_defined;   /// @endcond   typedef implementation_defined type;};#ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED#if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)template<class T, class O1, class O2, class O3, class O4, class O5, class O6, class O7, class O8, class O9, class O10>#elsetemplate<class T, class ...Options>#endifclass unordered_multiset   :  public make_unordered_multiset<T,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type{   typedef typename make_unordered_multiset      <T,         #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)         O1, O2, O3, O4, O5, O6, O7, O8, O9, O10         #else         Options...         #endif      >::type   Base;   //Assert if passed value traits are compatible with the type   BOOST_STATIC_ASSERT((detail::is_same<typename Base::value_traits::value_type, T>::value));   BOOST_MOVABLE_BUT_NOT_COPYABLE(unordered_multiset)   public:   typedef typename Base::value_traits       value_traits;   typedef typename Base::bucket_traits      bucket_traits;   typedef typename Base::iterator           iterator;   typedef typename Base::const_iterator     const_iterator;   typedef typename Base::bucket_ptr         bucket_ptr;   typedef typename Base::size_type          size_type;   typedef typename Base::hasher             hasher;   typedef typename Base::key_equal          key_equal;   explicit unordered_multiset( const bucket_traits &b_traits                              , const hasher & hash_func = hasher()                              , const key_equal &equal_func = key_equal()                              , const value_traits &v_traits = value_traits())      :  Base(b_traits, hash_func, equal_func, v_traits)   {}   template<class Iterator>   unordered_multiset( Iterator b                     , Iterator e                     , const bucket_traits &b_traits                     , const hasher & hash_func = hasher()                     , const key_equal &equal_func = key_equal()                     , const value_traits &v_traits = value_traits())      :  Base(b, e, b_traits, hash_func, equal_func, v_traits)   {}   unordered_multiset(BOOST_RV_REF(unordered_multiset) x)      :  Base(::boost::move(static_cast<Base&>(x)))   {}   unordered_multiset& operator=(BOOST_RV_REF(unordered_multiset) x)   {  return static_cast<unordered_multiset&>(this->Base::operator=(::boost::move(static_cast<Base&>(x))));  }};#endif} //namespace intrusive} //namespace boost#include <boost/intrusive/detail/config_end.hpp>#endif //BOOST_INTRUSIVE_UNORDERED_SET_HPP
 |