treap.hpp 54 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284
  1. /////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2008-2013
  4. //
  5. // Distributed under the Boost Software License, Version 1.0.
  6. // (See accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. //
  9. // See http://www.boost.org/libs/intrusive for documentation.
  10. //
  11. /////////////////////////////////////////////////////////////////////////////
  12. #ifndef BOOST_INTRUSIVE_TREAP_HPP
  13. #define BOOST_INTRUSIVE_TREAP_HPP
  14. #include <boost/intrusive/detail/config_begin.hpp>
  15. #include <boost/intrusive/intrusive_fwd.hpp>
  16. #include <boost/intrusive/detail/assert.hpp>
  17. #include <boost/intrusive/bs_set_hook.hpp>
  18. #include <boost/intrusive/bstree.hpp>
  19. #include <boost/intrusive/detail/tree_node.hpp>
  20. #include <boost/intrusive/detail/ebo_functor_holder.hpp>
  21. #include <boost/intrusive/pointer_traits.hpp>
  22. #include <boost/intrusive/detail/get_value_traits.hpp>
  23. #include <boost/intrusive/detail/mpl.hpp>
  24. #include <boost/intrusive/treap_algorithms.hpp>
  25. #include <boost/intrusive/link_mode.hpp>
  26. #include <boost/intrusive/priority_compare.hpp>
  27. #include <boost/intrusive/detail/node_cloner_disposer.hpp>
  28. #include <boost/intrusive/detail/key_nodeptr_comp.hpp>
  29. #include <boost/static_assert.hpp>
  30. #include <boost/move/utility_core.hpp>
  31. #include <boost/move/adl_move_swap.hpp>
  32. #include <cstddef>
  33. #include <boost/intrusive/detail/minimal_less_equal_header.hpp>
  34. #include <boost/intrusive/detail/minimal_pair_header.hpp> //std::pair
  35. #if defined(BOOST_HAS_PRAGMA_ONCE)
  36. # pragma once
  37. #endif
  38. namespace boost {
  39. namespace intrusive {
  40. /// @cond
  41. struct treap_defaults
  42. : bstree_defaults
  43. {
  44. typedef void priority;
  45. };
  46. /// @endcond
  47. //! The class template treap is an intrusive treap container that
  48. //! is used to construct intrusive set and multiset containers. The no-throw
  49. //! guarantee holds only, if the key_compare object and priority_compare object
  50. //! don't throw.
  51. //!
  52. //! The template parameter \c T is the type to be managed by the container.
  53. //! The user can specify additional options and if no options are provided
  54. //! default options are used.
  55. //!
  56. //! The container supports the following options:
  57. //! \c base_hook<>/member_hook<>/value_traits<>,
  58. //! \c constant_time_size<>, \c size_type<>,
  59. //! \c compare<> and \c priority_compare<>
  60. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)
  61. template<class T, class ...Options>
  62. #else
  63. template<class ValueTraits, class VoidOrKeyOfValue, class VoidOrKeyComp, class VoidOrPrioComp, class SizeType, bool ConstantTimeSize, typename HeaderHolder>
  64. #endif
  65. class treap_impl
  66. /// @cond
  67. : public bstree_impl<ValueTraits, VoidOrKeyOfValue, VoidOrKeyComp, SizeType, ConstantTimeSize, BsTreeAlgorithms, HeaderHolder>
  68. //Use public inheritance to avoid MSVC bugs with closures
  69. , public detail::ebo_functor_holder
  70. < typename get_prio
  71. < VoidOrPrioComp
  72. , typename bstree_impl
  73. <ValueTraits, VoidOrKeyOfValue, VoidOrKeyComp, SizeType, ConstantTimeSize, BsTreeAlgorithms, HeaderHolder>::key_type>::type
  74. >
  75. /// @endcond
  76. {
  77. public:
  78. typedef ValueTraits value_traits;
  79. /// @cond
  80. typedef bstree_impl< ValueTraits, VoidOrKeyOfValue, VoidOrKeyComp, SizeType
  81. , ConstantTimeSize, BsTreeAlgorithms
  82. , HeaderHolder> tree_type;
  83. typedef tree_type implementation_defined;
  84. typedef get_prio
  85. < VoidOrPrioComp
  86. , typename tree_type::key_type> get_prio_type;
  87. typedef detail::ebo_functor_holder
  88. <typename get_prio_type::type> prio_base;
  89. /// @endcond
  90. typedef typename implementation_defined::pointer pointer;
  91. typedef typename implementation_defined::const_pointer const_pointer;
  92. typedef typename implementation_defined::value_type value_type;
  93. typedef typename implementation_defined::key_type key_type;
  94. typedef typename implementation_defined::key_of_value key_of_value;
  95. typedef typename implementation_defined::reference reference;
  96. typedef typename implementation_defined::const_reference const_reference;
  97. typedef typename implementation_defined::difference_type difference_type;
  98. typedef typename implementation_defined::size_type size_type;
  99. typedef typename implementation_defined::value_compare value_compare;
  100. typedef typename implementation_defined::key_compare key_compare;
  101. typedef typename implementation_defined::iterator iterator;
  102. typedef typename implementation_defined::const_iterator const_iterator;
  103. typedef typename implementation_defined::reverse_iterator reverse_iterator;
  104. typedef typename implementation_defined::const_reverse_iterator const_reverse_iterator;
  105. typedef typename implementation_defined::node_traits node_traits;
  106. typedef typename implementation_defined::node node;
  107. typedef typename implementation_defined::node_ptr node_ptr;
  108. typedef typename implementation_defined::const_node_ptr const_node_ptr;
  109. typedef BOOST_INTRUSIVE_IMPDEF(treap_algorithms<node_traits>) node_algorithms;
  110. typedef BOOST_INTRUSIVE_IMPDEF(typename get_prio_type::type) priority_compare;
  111. static const bool constant_time_size = implementation_defined::constant_time_size;
  112. static const bool stateful_value_traits = implementation_defined::stateful_value_traits;
  113. static const bool safemode_or_autounlink = is_safe_autounlink<value_traits::link_mode>::value;
  114. typedef detail::key_nodeptr_comp<priority_compare, value_traits, key_of_value> key_node_prio_comp_t;
  115. template<class KeyPrioComp>
  116. detail::key_nodeptr_comp<KeyPrioComp, value_traits, key_of_value> key_node_prio_comp(KeyPrioComp keypriocomp) const
  117. { return detail::key_nodeptr_comp<KeyPrioComp, value_traits, key_of_value>(keypriocomp, &this->get_value_traits()); }
  118. /// @cond
  119. private:
  120. //noncopyable
  121. BOOST_MOVABLE_BUT_NOT_COPYABLE(treap_impl)
  122. const priority_compare &priv_pcomp() const
  123. { return static_cast<const prio_base&>(*this).get(); }
  124. priority_compare &priv_pcomp()
  125. { return static_cast<prio_base&>(*this).get(); }
  126. /// @endcond
  127. public:
  128. typedef typename node_algorithms::insert_commit_data insert_commit_data;
  129. //! <b>Effects</b>: Constructs an empty container.
  130. //!
  131. //! <b>Complexity</b>: Constant.
  132. //!
  133. //! <b>Throws</b>: If value_traits::node_traits::node
  134. //! constructor throws (this does not happen with predefined Boost.Intrusive hooks)
  135. //! or the copy constructor of the value_compare/priority_compare objects throw. Basic guarantee.
  136. treap_impl()
  137. : tree_type(), prio_base(priority_compare())
  138. {}
  139. //! <b>Effects</b>: Constructs an empty container.
  140. //!
  141. //! <b>Complexity</b>: Constant.
  142. //!
  143. //! <b>Throws</b>: If value_traits::node_traits::node
  144. //! constructor throws (this does not happen with predefined Boost.Intrusive hooks)
  145. //! or the copy constructor of the value_compare/priority_compare objects throw. Basic guarantee.
  146. explicit treap_impl( const key_compare &cmp
  147. , const priority_compare &pcmp = priority_compare()
  148. , const value_traits &v_traits = value_traits())
  149. : tree_type(cmp, v_traits), prio_base(pcmp)
  150. {}
  151. //! <b>Requires</b>: Dereferencing iterator must yield an lvalue of type value_type.
  152. //! cmp must be a comparison function that induces a strict weak ordering.
  153. //!
  154. //! <b>Effects</b>: Constructs an empty container and inserts elements from
  155. //! [b, e).
  156. //!
  157. //! <b>Complexity</b>: Linear in N if [b, e) is already sorted using
  158. //! comp and otherwise N * log N, where N is the distance between first and last.
  159. //!
  160. //! <b>Throws</b>: If value_traits::node_traits::node
  161. //! constructor throws (this does not happen with predefined Boost.Intrusive hooks)
  162. //! or the copy constructor/operator() of the key_compare/priority_compare objects
  163. //! throw. Basic guarantee.
  164. template<class Iterator>
  165. treap_impl( bool unique, Iterator b, Iterator e
  166. , const key_compare &cmp = key_compare()
  167. , const priority_compare &pcmp = priority_compare()
  168. , const value_traits &v_traits = value_traits())
  169. : tree_type(cmp, v_traits), prio_base(pcmp)
  170. {
  171. if(unique)
  172. this->insert_unique(b, e);
  173. else
  174. this->insert_equal(b, e);
  175. }
  176. //! @copydoc ::boost::intrusive::bstree::bstree(bstree &&)
  177. treap_impl(BOOST_RV_REF(treap_impl) x)
  178. : tree_type(BOOST_MOVE_BASE(tree_type, x))
  179. , prio_base(::boost::move(x.priv_pcomp()))
  180. {}
  181. //! @copydoc ::boost::intrusive::bstree::operator=(bstree &&)
  182. treap_impl& operator=(BOOST_RV_REF(treap_impl) x)
  183. { this->swap(x); return *this; }
  184. #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  185. //! @copydoc ::boost::intrusive::bstree::~bstree()
  186. ~treap_impl();
  187. //! @copydoc ::boost::intrusive::bstree::begin()
  188. iterator begin();
  189. //! @copydoc ::boost::intrusive::bstree::begin()const
  190. const_iterator begin() const;
  191. //! @copydoc ::boost::intrusive::bstree::cbegin()const
  192. const_iterator cbegin() const;
  193. //! @copydoc ::boost::intrusive::bstree::end()
  194. iterator end();
  195. //! @copydoc ::boost::intrusive::bstree::end()const
  196. const_iterator end() const;
  197. //! @copydoc ::boost::intrusive::bstree::cend()const
  198. const_iterator cend() const;
  199. #endif
  200. //! <b>Effects</b>: Returns an iterator pointing to the highest priority object of the treap.
  201. //!
  202. //! <b>Complexity</b>: Constant.
  203. //!
  204. //! <b>Throws</b>: Nothing.
  205. iterator top()
  206. { return this->tree_type::root(); }
  207. //! <b>Effects</b>: Returns a const_iterator pointing to the highest priority object of the treap..
  208. //!
  209. //! <b>Complexity</b>: Constant.
  210. //!
  211. //! <b>Throws</b>: Nothing.
  212. const_iterator top() const
  213. { return this->ctop(); }
  214. //! <b>Effects</b>: Returns a const_iterator pointing to the highest priority object of the treap..
  215. //!
  216. //! <b>Complexity</b>: Constant.
  217. //!
  218. //! <b>Throws</b>: Nothing.
  219. const_iterator ctop() const
  220. { return this->tree_type::root(); }
  221. #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  222. //! @copydoc ::boost::intrusive::bstree::rbegin()
  223. reverse_iterator rbegin();
  224. //! @copydoc ::boost::intrusive::bstree::rbegin()const
  225. const_reverse_iterator rbegin() const;
  226. //! @copydoc ::boost::intrusive::bstree::crbegin()const
  227. const_reverse_iterator crbegin() const;
  228. //! @copydoc ::boost::intrusive::bstree::rend()
  229. reverse_iterator rend();
  230. //! @copydoc ::boost::intrusive::bstree::rend()const
  231. const_reverse_iterator rend() const;
  232. //! @copydoc ::boost::intrusive::bstree::crend()const
  233. const_reverse_iterator crend() const;
  234. #endif
  235. //! <b>Effects</b>: Returns a reverse_iterator pointing to the highest priority object of the
  236. //! reversed treap.
  237. //!
  238. //! <b>Complexity</b>: Constant.
  239. //!
  240. //! <b>Throws</b>: Nothing.
  241. reverse_iterator rtop()
  242. { return reverse_iterator(this->top()); }
  243. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the highest priority objec
  244. //! of the reversed treap.
  245. //!
  246. //! <b>Complexity</b>: Constant.
  247. //!
  248. //! <b>Throws</b>: Nothing.
  249. const_reverse_iterator rtop() const
  250. { return const_reverse_iterator(this->top()); }
  251. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the highest priority object
  252. //! of the reversed treap.
  253. //!
  254. //! <b>Complexity</b>: Constant.
  255. //!
  256. //! <b>Throws</b>: Nothing.
  257. const_reverse_iterator crtop() const
  258. { return const_reverse_iterator(this->top()); }
  259. #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  260. //! @copydoc ::boost::intrusive::bstree::container_from_end_iterator(iterator)
  261. static treap_impl &container_from_end_iterator(iterator end_iterator);
  262. //! @copydoc ::boost::intrusive::bstree::container_from_end_iterator(const_iterator)
  263. static const treap_impl &container_from_end_iterator(const_iterator end_iterator);
  264. //! @copydoc ::boost::intrusive::bstree::container_from_iterator(iterator)
  265. static treap_impl &container_from_iterator(iterator it);
  266. //! @copydoc ::boost::intrusive::bstree::container_from_iterator(const_iterator)
  267. static const treap_impl &container_from_iterator(const_iterator it);
  268. //! @copydoc ::boost::intrusive::bstree::key_comp()const
  269. key_compare key_comp() const;
  270. //! @copydoc ::boost::intrusive::bstree::value_comp()const
  271. value_compare value_comp() const;
  272. //! @copydoc ::boost::intrusive::bstree::empty()const
  273. bool empty() const;
  274. //! @copydoc ::boost::intrusive::bstree::size()const
  275. size_type size() const;
  276. #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  277. //! <b>Effects</b>: Returns the priority_compare object used by the container.
  278. //!
  279. //! <b>Complexity</b>: Constant.
  280. //!
  281. //! <b>Throws</b>: If priority_compare copy-constructor throws.
  282. priority_compare priority_comp() const
  283. { return this->priv_pcomp(); }
  284. //! <b>Effects</b>: Swaps the contents of two treaps.
  285. //!
  286. //! <b>Complexity</b>: Constant.
  287. //!
  288. //! <b>Throws</b>: If the comparison functor's swap call throws.
  289. void swap(treap_impl& other)
  290. {
  291. tree_type::swap(other);
  292. //This can throw
  293. ::boost::adl_move_swap(this->priv_pcomp(), other.priv_pcomp());
  294. }
  295. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  296. //! Cloner should yield to nodes equivalent to the original nodes.
  297. //!
  298. //! <b>Effects</b>: Erases all the elements from *this
  299. //! calling Disposer::operator()(pointer), clones all the
  300. //! elements from src calling Cloner::operator()(const_reference )
  301. //! and inserts them on *this. Copies the predicate from the source container.
  302. //!
  303. //! If cloner throws, all cloned elements are unlinked and disposed
  304. //! calling Disposer::operator()(pointer).
  305. //!
  306. //! <b>Complexity</b>: Linear to erased plus inserted elements.
  307. //!
  308. //! <b>Throws</b>: If cloner throws or predicate copy assignment throws. Basic guarantee.
  309. template <class Cloner, class Disposer>
  310. void clone_from(const treap_impl &src, Cloner cloner, Disposer disposer)
  311. {
  312. tree_type::clone_from(src, cloner, disposer);
  313. this->priv_pcomp() = src.priv_pcomp();
  314. }
  315. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  316. //! Cloner should yield to nodes equivalent to the original nodes.
  317. //!
  318. //! <b>Effects</b>: Erases all the elements from *this
  319. //! calling Disposer::operator()(pointer), clones all the
  320. //! elements from src calling Cloner::operator()(reference)
  321. //! and inserts them on *this. Copies the predicate from the source container.
  322. //!
  323. //! If cloner throws, all cloned elements are unlinked and disposed
  324. //! calling Disposer::operator()(pointer).
  325. //!
  326. //! <b>Complexity</b>: Linear to erased plus inserted elements.
  327. //!
  328. //! <b>Throws</b>: If cloner throws or predicate copy assignment throws. Basic guarantee.
  329. template <class Cloner, class Disposer>
  330. void clone_from(BOOST_RV_REF(treap_impl) src, Cloner cloner, Disposer disposer)
  331. {
  332. tree_type::clone_from(BOOST_MOVE_BASE(tree_type, src), cloner, disposer);
  333. this->priv_pcomp() = ::boost::move(src.priv_pcomp());
  334. }
  335. //! <b>Requires</b>: value must be an lvalue
  336. //!
  337. //! <b>Effects</b>: Inserts value into the container before the upper bound.
  338. //!
  339. //! <b>Complexity</b>: Average complexity for insert element is at
  340. //! most logarithmic.
  341. //!
  342. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw. Strong guarantee.
  343. //!
  344. //! <b>Note</b>: Does not affect the validity of iterators and references.
  345. //! No copy-constructors are called.
  346. iterator insert_equal(reference value)
  347. {
  348. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  349. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  350. iterator ret
  351. ( node_algorithms::insert_equal_upper_bound
  352. ( this->tree_type::header_ptr()
  353. , to_insert
  354. , this->key_node_comp(this->key_comp())
  355. , this->key_node_prio_comp(this->priv_pcomp()))
  356. , this->priv_value_traits_ptr());
  357. this->tree_type::sz_traits().increment();
  358. return ret;
  359. }
  360. //! <b>Requires</b>: value must be an lvalue, and "hint" must be
  361. //! a valid iterator.
  362. //!
  363. //! <b>Effects</b>: Inserts x into the container, using "hint" as a hint to
  364. //! where it will be inserted. If "hint" is the upper_bound
  365. //! the insertion takes constant time (two comparisons in the worst case)
  366. //!
  367. //! <b>Complexity</b>: Logarithmic in general, but it is amortized
  368. //! constant time if t is inserted immediately before hint.
  369. //!
  370. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw. Strong guarantee.
  371. //!
  372. //! <b>Note</b>: Does not affect the validity of iterators and references.
  373. //! No copy-constructors are called.
  374. iterator insert_equal(const_iterator hint, reference value)
  375. {
  376. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  377. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  378. iterator ret
  379. (node_algorithms::insert_equal
  380. ( this->tree_type::header_ptr()
  381. , hint.pointed_node()
  382. , to_insert
  383. , this->key_node_comp(this->key_comp())
  384. , this->key_node_prio_comp(this->priv_pcomp()))
  385. , this->priv_value_traits_ptr());
  386. this->tree_type::sz_traits().increment();
  387. return ret;
  388. }
  389. //! <b>Requires</b>: Dereferencing iterator must yield an lvalue
  390. //! of type value_type.
  391. //!
  392. //! <b>Effects</b>: Inserts a each element of a range into the container
  393. //! before the upper bound of the key of each element.
  394. //!
  395. //! <b>Complexity</b>: Insert range is in general O(N * log(N)), where N is the
  396. //! size of the range. However, it is linear in N if the range is already sorted
  397. //! by key_comp().
  398. //!
  399. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw.
  400. //! Strong guarantee.
  401. //!
  402. //! <b>Note</b>: Does not affect the validity of iterators and references.
  403. //! No copy-constructors are called.
  404. template<class Iterator>
  405. void insert_equal(Iterator b, Iterator e)
  406. {
  407. iterator iend(this->end());
  408. for (; b != e; ++b)
  409. this->insert_equal(iend, *b);
  410. }
  411. //! <b>Requires</b>: value must be an lvalue
  412. //!
  413. //! <b>Effects</b>: Inserts value into the container if the value
  414. //! is not already present.
  415. //!
  416. //! <b>Complexity</b>: Average complexity for insert element is at
  417. //! most logarithmic.
  418. //!
  419. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw.
  420. //! Strong guarantee.
  421. //!
  422. //! <b>Note</b>: Does not affect the validity of iterators and references.
  423. //! No copy-constructors are called.
  424. std::pair<iterator, bool> insert_unique(reference value)
  425. {
  426. insert_commit_data commit_data;
  427. std::pair<iterator, bool> ret = this->insert_unique_check(key_of_value()(value), commit_data);
  428. if(!ret.second)
  429. return ret;
  430. return std::pair<iterator, bool> (this->insert_unique_commit(value, commit_data), true);
  431. }
  432. //! <b>Requires</b>: value must be an lvalue, and "hint" must be
  433. //! a valid iterator
  434. //!
  435. //! <b>Effects</b>: Tries to insert x into the container, using "hint" as a hint
  436. //! to where it will be inserted.
  437. //!
  438. //! <b>Complexity</b>: Logarithmic in general, but it is amortized
  439. //! constant time (two comparisons in the worst case)
  440. //! if t is inserted immediately before hint.
  441. //!
  442. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw.
  443. //! Strong guarantee.
  444. //!
  445. //! <b>Note</b>: Does not affect the validity of iterators and references.
  446. //! No copy-constructors are called.
  447. iterator insert_unique(const_iterator hint, reference value)
  448. {
  449. insert_commit_data commit_data;
  450. std::pair<iterator, bool> ret = this->insert_unique_check(hint, key_of_value()(value), commit_data);
  451. if(!ret.second)
  452. return ret.first;
  453. return this->insert_unique_commit(value, commit_data);
  454. }
  455. //! <b>Requires</b>: Dereferencing iterator must yield an lvalue
  456. //! of type value_type.
  457. //!
  458. //! <b>Effects</b>: Tries to insert each element of a range into the container.
  459. //!
  460. //! <b>Complexity</b>: Insert range is in general O(N * log(N)), where N is the
  461. //! size of the range. However, it is linear in N if the range is already sorted
  462. //! by key_comp().
  463. //!
  464. //! <b>Throws</b>: If the internal key_compare or priority_compare functions throw.
  465. //! Strong guarantee.
  466. //!
  467. //! <b>Note</b>: Does not affect the validity of iterators and references.
  468. //! No copy-constructors are called.
  469. template<class Iterator>
  470. void insert_unique(Iterator b, Iterator e)
  471. {
  472. if(this->empty()){
  473. iterator iend(this->end());
  474. for (; b != e; ++b)
  475. this->insert_unique(iend, *b);
  476. }
  477. else{
  478. for (; b != e; ++b)
  479. this->insert_unique(*b);
  480. }
  481. }
  482. //! <b>Effects</b>: Checks if a value can be inserted in the container, using
  483. //! a user provided key instead of the value itself.
  484. //!
  485. //! <b>Returns</b>: If there is an equivalent value
  486. //! returns a pair containing an iterator to the already present value
  487. //! and false. If the value can be inserted returns true in the returned
  488. //! pair boolean and fills "commit_data" that is meant to be used with
  489. //! the "insert_commit" function.
  490. //!
  491. //! <b>Complexity</b>: Average complexity is at most logarithmic.
  492. //!
  493. //! <b>Throws</b>: If the comparison or predicate functions throw. Strong guarantee.
  494. //!
  495. //! <b>Notes</b>: This function is used to improve performance when constructing
  496. //! a value_type is expensive: if there is an equivalent value
  497. //! the constructed object must be discarded. Many times, the part of the
  498. //! node that is used to impose the order is much cheaper to construct
  499. //! than the value_type and this function offers the possibility to use that
  500. //! part to check if the insertion will be successful.
  501. //!
  502. //! If the check is successful, the user can construct the value_type and use
  503. //! "insert_commit" to insert the object in constant-time. This gives a total
  504. //! logarithmic complexity to the insertion: check(O(log(N)) + commit(O(1)).
  505. //!
  506. //! "commit_data" remains valid for a subsequent "insert_commit" only if no more
  507. //! objects are inserted or erased from the container.
  508. std::pair<iterator, bool> insert_unique_check
  509. ( const key_type &key, insert_commit_data &commit_data)
  510. { return this->insert_unique_check(key, this->key_comp(), this->priv_pcomp(), commit_data); }
  511. //! <b>Effects</b>: Checks if a value can be inserted in the container, using
  512. //! a user provided key instead of the value itself, using "hint"
  513. //! as a hint to where it will be inserted.
  514. //!
  515. //! <b>Returns</b>: If there is an equivalent value
  516. //! returns a pair containing an iterator to the already present value
  517. //! and false. If the value can be inserted returns true in the returned
  518. //! pair boolean and fills "commit_data" that is meant to be used with
  519. //! the "insert_commit" function.
  520. //!
  521. //! <b>Complexity</b>: Logarithmic in general, but it's amortized
  522. //! constant time if t is inserted immediately before hint.
  523. //!
  524. //! <b>Throws</b>: If the comparison or predicate functions throw. Strong guarantee.
  525. //!
  526. //! <b>Notes</b>: This function is used to improve performance when constructing
  527. //! a value_type is expensive: if there is an equivalent value
  528. //! the constructed object must be discarded. Many times, the part of the
  529. //! constructing that is used to impose the order is much cheaper to construct
  530. //! than the value_type and this function offers the possibility to use that key
  531. //! to check if the insertion will be successful.
  532. //!
  533. //! If the check is successful, the user can construct the value_type and use
  534. //! "insert_commit" to insert the object in constant-time. This can give a total
  535. //! constant-time complexity to the insertion: check(O(1)) + commit(O(1)).
  536. //!
  537. //! "commit_data" remains valid for a subsequent "insert_commit" only if no more
  538. //! objects are inserted or erased from the container.
  539. std::pair<iterator, bool> insert_unique_check
  540. ( const_iterator hint, const key_type &key, insert_commit_data &commit_data)
  541. { return this->insert_unique_check(hint, key, this->key_comp(), this->priv_pcomp(), commit_data); }
  542. //! <b>Requires</b>: comp must be a comparison function that induces
  543. //! the same strict weak ordering as key_compare.
  544. //! key_value_pcomp must be a comparison function that induces
  545. //! the same strict weak ordering as priority_compare. The difference is that
  546. //! key_value_pcomp and comp compare an arbitrary key with the contained values.
  547. //!
  548. //! <b>Effects</b>: Checks if a value can be inserted in the container, using
  549. //! a user provided key instead of the value itself.
  550. //!
  551. //! <b>Returns</b>: If there is an equivalent value
  552. //! returns a pair containing an iterator to the already present value
  553. //! and false. If the value can be inserted returns true in the returned
  554. //! pair boolean and fills "commit_data" that is meant to be used with
  555. //! the "insert_commit" function.
  556. //!
  557. //! <b>Complexity</b>: Average complexity is at most logarithmic.
  558. //!
  559. //! <b>Throws</b>: If the comp or key_value_pcomp
  560. //! ordering functions throw. Strong guarantee.
  561. //!
  562. //! <b>Notes</b>: This function is used to improve performance when constructing
  563. //! a value_type is expensive: if there is an equivalent value
  564. //! the constructed object must be discarded. Many times, the part of the
  565. //! node that is used to impose the order is much cheaper to construct
  566. //! than the value_type and this function offers the possibility to use that
  567. //! part to check if the insertion will be successful.
  568. //!
  569. //! If the check is successful, the user can construct the value_type and use
  570. //! "insert_commit" to insert the object in constant-time. This gives a total
  571. //! logarithmic complexity to the insertion: check(O(log(N)) + commit(O(1)).
  572. //!
  573. //! "commit_data" remains valid for a subsequent "insert_commit" only if no more
  574. //! objects are inserted or erased from the container.
  575. template<class KeyType, class KeyTypeKeyCompare, class KeyValuePrioCompare>
  576. BOOST_INTRUSIVE_DOC1ST(std::pair<iterator BOOST_INTRUSIVE_I bool>
  577. , typename detail::disable_if_convertible
  578. <KeyType BOOST_INTRUSIVE_I const_iterator BOOST_INTRUSIVE_I
  579. std::pair<iterator BOOST_INTRUSIVE_I bool> >::type)
  580. insert_unique_check
  581. ( const KeyType &key, KeyTypeKeyCompare comp
  582. , KeyValuePrioCompare key_value_pcomp, insert_commit_data &commit_data)
  583. {
  584. std::pair<node_ptr, bool> const ret =
  585. (node_algorithms::insert_unique_check
  586. ( this->tree_type::header_ptr(), key
  587. , this->key_node_comp(comp), this->key_node_prio_comp(key_value_pcomp), commit_data));
  588. return std::pair<iterator, bool>(iterator(ret.first, this->priv_value_traits_ptr()), ret.second);
  589. }
  590. //! <b>Requires</b>: comp must be a comparison function that induces
  591. //! the same strict weak ordering as key_compare.
  592. //! key_value_pcomp must be a comparison function that induces
  593. //! the same strict weak ordering as priority_compare. The difference is that
  594. //! key_value_pcomp and comp compare an arbitrary key with the contained values.
  595. //!
  596. //! <b>Effects</b>: Checks if a value can be inserted in the container, using
  597. //! a user provided key instead of the value itself, using "hint"
  598. //! as a hint to where it will be inserted.
  599. //!
  600. //! <b>Returns</b>: If there is an equivalent value
  601. //! returns a pair containing an iterator to the already present value
  602. //! and false. If the value can be inserted returns true in the returned
  603. //! pair boolean and fills "commit_data" that is meant to be used with
  604. //! the "insert_commit" function.
  605. //!
  606. //! <b>Complexity</b>: Logarithmic in general, but it's amortized
  607. //! constant time if t is inserted immediately before hint.
  608. //!
  609. //! <b>Throws</b>: If the comp or key_value_pcomp
  610. //! ordering functions throw. Strong guarantee.
  611. //!
  612. //! <b>Notes</b>: This function is used to improve performance when constructing
  613. //! a value_type is expensive: if there is an equivalent value
  614. //! the constructed object must be discarded. Many times, the part of the
  615. //! constructing that is used to impose the order is much cheaper to construct
  616. //! than the value_type and this function offers the possibility to use that key
  617. //! to check if the insertion will be successful.
  618. //!
  619. //! If the check is successful, the user can construct the value_type and use
  620. //! "insert_commit" to insert the object in constant-time. This can give a total
  621. //! constant-time complexity to the insertion: check(O(1)) + commit(O(1)).
  622. //!
  623. //! "commit_data" remains valid for a subsequent "insert_commit" only if no more
  624. //! objects are inserted or erased from the container.
  625. template<class KeyType, class KeyTypeKeyCompare, class KeyValuePrioCompare>
  626. std::pair<iterator, bool> insert_unique_check
  627. ( const_iterator hint, const KeyType &key
  628. , KeyTypeKeyCompare comp
  629. , KeyValuePrioCompare key_value_pcomp
  630. , insert_commit_data &commit_data)
  631. {
  632. std::pair<node_ptr, bool> const ret =
  633. (node_algorithms::insert_unique_check
  634. ( this->tree_type::header_ptr(), hint.pointed_node(), key
  635. , this->key_node_comp(comp), this->key_node_prio_comp(key_value_pcomp), commit_data));
  636. return std::pair<iterator, bool>(iterator(ret.first, this->priv_value_traits_ptr()), ret.second);
  637. }
  638. //! <b>Requires</b>: value must be an lvalue of type value_type. commit_data
  639. //! must have been obtained from a previous call to "insert_check".
  640. //! No objects should have been inserted or erased from the container between
  641. //! the "insert_check" that filled "commit_data" and the call to "insert_commit".
  642. //!
  643. //! <b>Effects</b>: Inserts the value in the avl_set using the information obtained
  644. //! from the "commit_data" that a previous "insert_check" filled.
  645. //!
  646. //! <b>Returns</b>: An iterator to the newly inserted object.
  647. //!
  648. //! <b>Complexity</b>: Constant time.
  649. //!
  650. //! <b>Throws</b>: Nothing
  651. //!
  652. //! <b>Notes</b>: This function has only sense if a "insert_check" has been
  653. //! previously executed to fill "commit_data". No value should be inserted or
  654. //! erased between the "insert_check" and "insert_commit" calls.
  655. iterator insert_unique_commit(reference value, const insert_commit_data &commit_data)
  656. {
  657. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  658. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  659. node_algorithms::insert_unique_commit(this->tree_type::header_ptr(), to_insert, commit_data);
  660. this->tree_type::sz_traits().increment();
  661. return iterator(to_insert, this->priv_value_traits_ptr());
  662. }
  663. //! <b>Requires</b>: value must be an lvalue, "pos" must be
  664. //! a valid iterator (or end) and must be the succesor of value
  665. //! once inserted according to the predicate
  666. //!
  667. //! <b>Effects</b>: Inserts x into the container before "pos".
  668. //!
  669. //! <b>Complexity</b>: Constant time.
  670. //!
  671. //! <b>Throws</b>: If the internal priority_compare function throws. Strong guarantee.
  672. //!
  673. //! <b>Note</b>: This function does not check preconditions so if "pos" is not
  674. //! the successor of "value" container ordering invariant will be broken.
  675. //! This is a low-level function to be used only for performance reasons
  676. //! by advanced users.
  677. iterator insert_before(const_iterator pos, reference value)
  678. {
  679. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  680. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  681. iterator ret
  682. ( node_algorithms::insert_before
  683. ( this->tree_type::header_ptr()
  684. , pos.pointed_node()
  685. , to_insert
  686. , this->key_node_prio_comp(this->priv_pcomp())
  687. )
  688. , this->priv_value_traits_ptr());
  689. this->tree_type::sz_traits().increment();
  690. return ret;
  691. }
  692. //! <b>Requires</b>: value must be an lvalue, and it must be no less
  693. //! than the greatest inserted key
  694. //!
  695. //! <b>Effects</b>: Inserts x into the container in the last position.
  696. //!
  697. //! <b>Complexity</b>: Constant time.
  698. //!
  699. //! <b>Throws</b>: If the internal priority_compare function throws. Strong guarantee.
  700. //!
  701. //! <b>Note</b>: This function does not check preconditions so if value is
  702. //! less than the greatest inserted key container ordering invariant will be broken.
  703. //! This function is slightly more efficient than using "insert_before".
  704. //! This is a low-level function to be used only for performance reasons
  705. //! by advanced users.
  706. void push_back(reference value)
  707. {
  708. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  709. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  710. node_algorithms::push_back
  711. (this->tree_type::header_ptr(), to_insert, this->key_node_prio_comp(this->priv_pcomp()));
  712. this->tree_type::sz_traits().increment();
  713. }
  714. //! <b>Requires</b>: value must be an lvalue, and it must be no greater
  715. //! than the minimum inserted key
  716. //!
  717. //! <b>Effects</b>: Inserts x into the container in the first position.
  718. //!
  719. //! <b>Complexity</b>: Constant time.
  720. //!
  721. //! <b>Throws</b>: If the internal priority_compare function throws. Strong guarantee.
  722. //!
  723. //! <b>Note</b>: This function does not check preconditions so if value is
  724. //! greater than the minimum inserted key container ordering invariant will be broken.
  725. //! This function is slightly more efficient than using "insert_before".
  726. //! This is a low-level function to be used only for performance reasons
  727. //! by advanced users.
  728. void push_front(reference value)
  729. {
  730. node_ptr to_insert(this->get_value_traits().to_node_ptr(value));
  731. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || node_algorithms::unique(to_insert));
  732. node_algorithms::push_front
  733. (this->tree_type::header_ptr(), to_insert, this->key_node_prio_comp(this->priv_pcomp()));
  734. this->tree_type::sz_traits().increment();
  735. }
  736. //! <b>Effects</b>: Erases the element pointed to by i.
  737. //!
  738. //! <b>Complexity</b>: Average complexity for erase element is constant time.
  739. //!
  740. //! <b>Throws</b>: if the internal priority_compare function throws. Strong guarantee.
  741. //!
  742. //! <b>Note</b>: Invalidates the iterators (but not the references)
  743. //! to the erased elements. No destructors are called.
  744. iterator erase(const_iterator i)
  745. {
  746. const_iterator ret(i);
  747. ++ret;
  748. node_ptr to_erase(i.pointed_node());
  749. BOOST_INTRUSIVE_SAFE_HOOK_DEFAULT_ASSERT(!safemode_or_autounlink || !node_algorithms::unique(to_erase));
  750. node_algorithms::erase
  751. (this->tree_type::header_ptr(), to_erase, this->key_node_prio_comp(this->priv_pcomp()));
  752. this->tree_type::sz_traits().decrement();
  753. if(safemode_or_autounlink)
  754. node_algorithms::init(to_erase);
  755. return ret.unconst();
  756. }
  757. //! <b>Effects</b>: Erases the range pointed to by b end e.
  758. //!
  759. //! <b>Complexity</b>: Average complexity for erase range is at most
  760. //! O(log(size() + N)), where N is the number of elements in the range.
  761. //!
  762. //! <b>Throws</b>: if the internal priority_compare function throws. Strong guarantee.
  763. //!
  764. //! <b>Note</b>: Invalidates the iterators (but not the references)
  765. //! to the erased elements. No destructors are called.
  766. iterator erase(const_iterator b, const_iterator e)
  767. { size_type n; return private_erase(b, e, n); }
  768. //! <b>Effects</b>: Erases all the elements with the given value.
  769. //!
  770. //! <b>Returns</b>: The number of erased elements.
  771. //!
  772. //! <b>Complexity</b>: O(log(size() + N).
  773. //!
  774. //! <b>Throws</b>: if the internal priority_compare function throws. Strong guarantee.
  775. //!
  776. //! <b>Note</b>: Invalidates the iterators (but not the references)
  777. //! to the erased elements. No destructors are called.
  778. size_type erase(const key_type &key)
  779. { return this->erase(key, this->key_comp()); }
  780. //! <b>Effects</b>: Erases all the elements with the given key.
  781. //! according to the comparison functor "comp".
  782. //!
  783. //! <b>Returns</b>: The number of erased elements.
  784. //!
  785. //! <b>Complexity</b>: O(log(size() + N).
  786. //!
  787. //! <b>Throws</b>: if the internal priority_compare function throws.
  788. //! Equivalent guarantee to <i>while(beg != end) erase(beg++);</i>
  789. //!
  790. //! <b>Note</b>: Invalidates the iterators (but not the references)
  791. //! to the erased elements. No destructors are called.
  792. template<class KeyType, class KeyTypeKeyCompare>
  793. BOOST_INTRUSIVE_DOC1ST(size_type
  794. , typename detail::disable_if_convertible<KeyTypeKeyCompare BOOST_INTRUSIVE_I const_iterator BOOST_INTRUSIVE_I size_type>::type)
  795. erase(const KeyType& key, KeyTypeKeyCompare comp)
  796. {
  797. std::pair<iterator,iterator> p = this->equal_range(key, comp);
  798. size_type n;
  799. private_erase(p.first, p.second, n);
  800. return n;
  801. }
  802. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  803. //!
  804. //! <b>Effects</b>: Erases the element pointed to by i.
  805. //! Disposer::operator()(pointer) is called for the removed element.
  806. //!
  807. //! <b>Complexity</b>: Average complexity for erase element is constant time.
  808. //!
  809. //! <b>Throws</b>: if the internal priority_compare function throws. Strong guarantee.
  810. //!
  811. //! <b>Note</b>: Invalidates the iterators
  812. //! to the erased elements.
  813. template<class Disposer>
  814. iterator erase_and_dispose(const_iterator i, Disposer disposer)
  815. {
  816. node_ptr to_erase(i.pointed_node());
  817. iterator ret(this->erase(i));
  818. disposer(this->get_value_traits().to_value_ptr(to_erase));
  819. return ret;
  820. }
  821. #if !defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED)
  822. template<class Disposer>
  823. iterator erase_and_dispose(iterator i, Disposer disposer)
  824. { return this->erase_and_dispose(const_iterator(i), disposer); }
  825. #endif
  826. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  827. //!
  828. //! <b>Effects</b>: Erases the range pointed to by b end e.
  829. //! Disposer::operator()(pointer) is called for the removed elements.
  830. //!
  831. //! <b>Complexity</b>: Average complexity for erase range is at most
  832. //! O(log(size() + N)), where N is the number of elements in the range.
  833. //!
  834. //! <b>Throws</b>: if the internal priority_compare function throws. Strong guarantee.
  835. //!
  836. //! <b>Note</b>: Invalidates the iterators
  837. //! to the erased elements.
  838. template<class Disposer>
  839. iterator erase_and_dispose(const_iterator b, const_iterator e, Disposer disposer)
  840. { size_type n; return private_erase(b, e, n, disposer); }
  841. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  842. //!
  843. //! <b>Effects</b>: Erases all the elements with the given value.
  844. //! Disposer::operator()(pointer) is called for the removed elements.
  845. //!
  846. //! <b>Returns</b>: The number of erased elements.
  847. //!
  848. //! <b>Complexity</b>: O(log(size() + N).
  849. //!
  850. //! <b>Throws</b>: if the priority_compare function throws then weak guarantee and heap invariants are broken.
  851. //! The safest thing would be to clear or destroy the container.
  852. //!
  853. //! <b>Note</b>: Invalidates the iterators (but not the references)
  854. //! to the erased elements. No destructors are called.
  855. template<class Disposer>
  856. size_type erase_and_dispose(const key_type &key, Disposer disposer)
  857. {
  858. std::pair<iterator,iterator> p = this->equal_range(key);
  859. size_type n;
  860. private_erase(p.first, p.second, n, disposer);
  861. return n;
  862. }
  863. //! <b>Requires</b>: Disposer::operator()(pointer) shouldn't throw.
  864. //!
  865. //! <b>Effects</b>: Erases all the elements with the given key.
  866. //! according to the comparison functor "comp".
  867. //! Disposer::operator()(pointer) is called for the removed elements.
  868. //!
  869. //! <b>Returns</b>: The number of erased elements.
  870. //!
  871. //! <b>Complexity</b>: O(log(size() + N).
  872. //!
  873. //! <b>Throws</b>: if the priority_compare function throws then weak guarantee and heap invariants are broken.
  874. //! The safest thing would be to clear or destroy the container.
  875. //!
  876. //! <b>Note</b>: Invalidates the iterators
  877. //! to the erased elements.
  878. template<class KeyType, class KeyTypeKeyCompare, class Disposer>
  879. BOOST_INTRUSIVE_DOC1ST(size_type
  880. , typename detail::disable_if_convertible<KeyTypeKeyCompare BOOST_INTRUSIVE_I const_iterator BOOST_INTRUSIVE_I size_type>::type)
  881. erase_and_dispose(const KeyType& key, KeyTypeKeyCompare comp, Disposer disposer)
  882. {
  883. std::pair<iterator,iterator> p = this->equal_range(key, comp);
  884. size_type n;
  885. private_erase(p.first, p.second, n, disposer);
  886. return n;
  887. }
  888. //! <b>Effects</b>: Erases all of the elements.
  889. //!
  890. //! <b>Complexity</b>: Linear to the number of elements on the container.
  891. //! if it's a safe-mode or auto-unlink value_type. Constant time otherwise.
  892. //!
  893. //! <b>Throws</b>: Nothing.
  894. //!
  895. //! <b>Note</b>: Invalidates the iterators (but not the references)
  896. //! to the erased elements. No destructors are called.
  897. void clear()
  898. { tree_type::clear(); }
  899. //! <b>Effects</b>: Erases all of the elements calling disposer(p) for
  900. //! each node to be erased.
  901. //! <b>Complexity</b>: Average complexity for is at most O(log(size() + N)),
  902. //! where N is the number of elements in the container.
  903. //!
  904. //! <b>Throws</b>: Nothing.
  905. //!
  906. //! <b>Note</b>: Invalidates the iterators (but not the references)
  907. //! to the erased elements. Calls N times to disposer functor.
  908. template<class Disposer>
  909. void clear_and_dispose(Disposer disposer)
  910. {
  911. node_algorithms::clear_and_dispose(this->tree_type::header_ptr()
  912. , detail::node_disposer<Disposer, value_traits, TreapAlgorithms>(disposer, &this->get_value_traits()));
  913. node_algorithms::init_header(this->tree_type::header_ptr());
  914. this->tree_type::sz_traits().set_size(0);
  915. }
  916. //! @copydoc ::boost::intrusive::bstree::check(ExtraChecker)const
  917. template <class ExtraChecker>
  918. void check(ExtraChecker extra_checker) const
  919. {
  920. typedef detail::key_nodeptr_comp<priority_compare, value_traits, key_of_value> nodeptr_prio_comp_t;
  921. tree_type::check(detail::treap_node_extra_checker
  922. <ValueTraits, nodeptr_prio_comp_t, ExtraChecker>
  923. (this->key_node_prio_comp(this->priv_pcomp()), extra_checker));
  924. }
  925. //! @copydoc ::boost::intrusive::bstree::check()const
  926. void check() const
  927. { check(detail::empty_node_checker<ValueTraits>()); }
  928. #ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  929. //! @copydoc ::boost::intrusive::bstree::count(const key_type &)const
  930. size_type count(const key_type &key) const;
  931. //! @copydoc ::boost::intrusive::bstree::count(const KeyType&,KeyTypeKeyCompare)const
  932. template<class KeyType, class KeyTypeKeyCompare>
  933. size_type count(const KeyType& key, KeyTypeKeyCompare comp) const;
  934. //! @copydoc ::boost::intrusive::bstree::lower_bound(const key_type &)
  935. iterator lower_bound(const key_type &key);
  936. //! @copydoc ::boost::intrusive::bstree::lower_bound(const KeyType&,KeyTypeKeyCompare)
  937. template<class KeyType, class KeyTypeKeyCompare>
  938. iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp);
  939. //! @copydoc ::boost::intrusive::bstree::lower_bound(const key_type &)const
  940. const_iterator lower_bound(const key_type &key) const;
  941. //! @copydoc ::boost::intrusive::bstree::lower_bound(const KeyType&,KeyTypeKeyCompare)const
  942. template<class KeyType, class KeyTypeKeyCompare>
  943. const_iterator lower_bound(const KeyType& key, KeyTypeKeyCompare comp) const;
  944. //! @copydoc ::boost::intrusive::bstree::upper_bound(const key_type &)
  945. iterator upper_bound(const key_type &key);
  946. //! @copydoc ::boost::intrusive::bstree::upper_bound(const KeyType&,KeyTypeKeyCompare)
  947. template<class KeyType, class KeyTypeKeyCompare>
  948. iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp);
  949. //! @copydoc ::boost::intrusive::bstree::upper_bound(const key_type &)const
  950. const_iterator upper_bound(const key_type &key) const;
  951. //! @copydoc ::boost::intrusive::bstree::upper_bound(const KeyType&,KeyTypeKeyCompare)const
  952. template<class KeyType, class KeyTypeKeyCompare>
  953. const_iterator upper_bound(const KeyType& key, KeyTypeKeyCompare comp) const;
  954. //! @copydoc ::boost::intrusive::bstree::find(const key_type &)
  955. iterator find(const key_type &key);
  956. //! @copydoc ::boost::intrusive::bstree::find(const KeyType&,KeyTypeKeyCompare)
  957. template<class KeyType, class KeyTypeKeyCompare>
  958. iterator find(const KeyType& key, KeyTypeKeyCompare comp);
  959. //! @copydoc ::boost::intrusive::bstree::find(const key_type &)const
  960. const_iterator find(const key_type &key) const;
  961. //! @copydoc ::boost::intrusive::bstree::find(const KeyType&,KeyTypeKeyCompare)const
  962. template<class KeyType, class KeyTypeKeyCompare>
  963. const_iterator find(const KeyType& key, KeyTypeKeyCompare comp) const;
  964. //! @copydoc ::boost::intrusive::bstree::equal_range(const key_type &)
  965. std::pair<iterator,iterator> equal_range(const key_type &key);
  966. //! @copydoc ::boost::intrusive::bstree::equal_range(const KeyType&,KeyTypeKeyCompare)
  967. template<class KeyType, class KeyTypeKeyCompare>
  968. std::pair<iterator,iterator> equal_range(const KeyType& key, KeyTypeKeyCompare comp);
  969. //! @copydoc ::boost::intrusive::bstree::equal_range(const key_type &)const
  970. std::pair<const_iterator, const_iterator>
  971. equal_range(const key_type &key) const;
  972. //! @copydoc ::boost::intrusive::bstree::equal_range(const KeyType&,KeyTypeKeyCompare)const
  973. template<class KeyType, class KeyTypeKeyCompare>
  974. std::pair<const_iterator, const_iterator>
  975. equal_range(const KeyType& key, KeyTypeKeyCompare comp) const;
  976. //! @copydoc ::boost::intrusive::bstree::bounded_range(const key_type &,const key_type &,bool,bool)
  977. std::pair<iterator,iterator> bounded_range
  978. (const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed);
  979. //! @copydoc ::boost::intrusive::bstree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool)
  980. template<class KeyType, class KeyTypeKeyCompare>
  981. std::pair<iterator,iterator> bounded_range
  982. (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed);
  983. //! @copydoc ::boost::intrusive::bstree::bounded_range(const key_type &,const key_type &,bool,bool)const
  984. std::pair<const_iterator, const_iterator>
  985. bounded_range(const key_type &lower_key, const key_type &upper_key, bool left_closed, bool right_closed) const;
  986. //! @copydoc ::boost::intrusive::bstree::bounded_range(const KeyType&,const KeyType&,KeyTypeKeyCompare,bool,bool)const
  987. template<class KeyType, class KeyTypeKeyCompare>
  988. std::pair<const_iterator, const_iterator> bounded_range
  989. (const KeyType& lower_key, const KeyType& upper_key, KeyTypeKeyCompare comp, bool left_closed, bool right_closed) const;
  990. //! @copydoc ::boost::intrusive::bstree::s_iterator_to(reference)
  991. static iterator s_iterator_to(reference value);
  992. //! @copydoc ::boost::intrusive::bstree::s_iterator_to(const_reference)
  993. static const_iterator s_iterator_to(const_reference value);
  994. //! @copydoc ::boost::intrusive::bstree::iterator_to(reference)
  995. iterator iterator_to(reference value);
  996. //! @copydoc ::boost::intrusive::bstree::iterator_to(const_reference)const
  997. const_iterator iterator_to(const_reference value) const;
  998. //! @copydoc ::boost::intrusive::bstree::init_node(reference)
  999. static void init_node(reference value);
  1000. //! @copydoc ::boost::intrusive::bstree::unlink_leftmost_without_rebalance
  1001. pointer unlink_leftmost_without_rebalance();
  1002. //! @copydoc ::boost::intrusive::bstree::replace_node
  1003. void replace_node(iterator replace_this, reference with_this);
  1004. //! @copydoc ::boost::intrusive::bstree::remove_node
  1005. void remove_node(reference value);
  1006. friend bool operator< (const treap_impl &x, const treap_impl &y);
  1007. friend bool operator==(const treap_impl &x, const treap_impl &y);
  1008. friend bool operator!= (const treap_impl &x, const treap_impl &y);
  1009. friend bool operator>(const treap_impl &x, const treap_impl &y);
  1010. friend bool operator<=(const treap_impl &x, const treap_impl &y);
  1011. friend bool operator>=(const treap_impl &x, const treap_impl &y);
  1012. friend void swap(treap_impl &x, treap_impl &y);
  1013. #endif //#ifdef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  1014. /// @cond
  1015. private:
  1016. template<class Disposer>
  1017. iterator private_erase(const_iterator b, const_iterator e, size_type &n, Disposer disposer)
  1018. {
  1019. for(n = 0; b != e; ++n)
  1020. this->erase_and_dispose(b++, disposer);
  1021. return b.unconst();
  1022. }
  1023. iterator private_erase(const_iterator b, const_iterator e, size_type &n)
  1024. {
  1025. for(n = 0; b != e; ++n)
  1026. this->erase(b++);
  1027. return b.unconst();
  1028. }
  1029. /// @endcond
  1030. };
  1031. //! Helper metafunction to define a \c treap that yields to the same type when the
  1032. //! same options (either explicitly or implicitly) are used.
  1033. #if defined(BOOST_INTRUSIVE_DOXYGEN_INVOKED) || defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  1034. template<class T, class ...Options>
  1035. #else
  1036. template<class T, class O1 = void, class O2 = void
  1037. , class O3 = void, class O4 = void
  1038. , class O5 = void, class O6 = void>
  1039. #endif
  1040. struct make_treap
  1041. {
  1042. typedef typename pack_options
  1043. < treap_defaults,
  1044. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  1045. O1, O2, O3, O4, O5, O6
  1046. #else
  1047. Options...
  1048. #endif
  1049. >::type packed_options;
  1050. typedef typename detail::get_value_traits
  1051. <T, typename packed_options::proto_value_traits>::type value_traits;
  1052. typedef treap_impl
  1053. < value_traits
  1054. , typename packed_options::key_of_value
  1055. , typename packed_options::compare
  1056. , typename packed_options::priority
  1057. , typename packed_options::size_type
  1058. , packed_options::constant_time_size
  1059. , typename packed_options::header_holder_type
  1060. > implementation_defined;
  1061. /// @endcond
  1062. typedef implementation_defined type;
  1063. };
  1064. #ifndef BOOST_INTRUSIVE_DOXYGEN_INVOKED
  1065. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  1066. template<class T, class O1, class O2, class O3, class O4, class O5, class O6>
  1067. #else
  1068. template<class T, class ...Options>
  1069. #endif
  1070. class treap
  1071. : public make_treap<T,
  1072. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  1073. O1, O2, O3, O4, O5, O6
  1074. #else
  1075. Options...
  1076. #endif
  1077. >::type
  1078. {
  1079. typedef typename make_treap
  1080. <T,
  1081. #if !defined(BOOST_INTRUSIVE_VARIADIC_TEMPLATES)
  1082. O1, O2, O3, O4, O5, O6
  1083. #else
  1084. Options...
  1085. #endif
  1086. >::type Base;
  1087. BOOST_MOVABLE_BUT_NOT_COPYABLE(treap)
  1088. public:
  1089. typedef typename Base::key_compare key_compare;
  1090. typedef typename Base::priority_compare priority_compare;
  1091. typedef typename Base::value_traits value_traits;
  1092. typedef typename Base::iterator iterator;
  1093. typedef typename Base::const_iterator const_iterator;
  1094. typedef typename Base::reverse_iterator reverse_iterator;
  1095. typedef typename Base::const_reverse_iterator const_reverse_iterator;
  1096. //Assert if passed value traits are compatible with the type
  1097. BOOST_STATIC_ASSERT((detail::is_same<typename value_traits::value_type, T>::value));
  1098. treap()
  1099. : Base()
  1100. {}
  1101. explicit treap( const key_compare &cmp
  1102. , const priority_compare &pcmp = priority_compare()
  1103. , const value_traits &v_traits = value_traits())
  1104. : Base(cmp, pcmp, v_traits)
  1105. {}
  1106. template<class Iterator>
  1107. treap( bool unique, Iterator b, Iterator e
  1108. , const key_compare &cmp = key_compare()
  1109. , const priority_compare &pcmp = priority_compare()
  1110. , const value_traits &v_traits = value_traits())
  1111. : Base(unique, b, e, cmp, pcmp, v_traits)
  1112. {}
  1113. treap(BOOST_RV_REF(treap) x)
  1114. : Base(BOOST_MOVE_BASE(Base, x))
  1115. {}
  1116. treap& operator=(BOOST_RV_REF(treap) x)
  1117. { return static_cast<treap&>(this->Base::operator=(BOOST_MOVE_BASE(Base, x))); }
  1118. template <class Cloner, class Disposer>
  1119. void clone_from(const treap &src, Cloner cloner, Disposer disposer)
  1120. { Base::clone_from(src, cloner, disposer); }
  1121. template <class Cloner, class Disposer>
  1122. void clone_from(BOOST_RV_REF(treap) src, Cloner cloner, Disposer disposer)
  1123. { Base::clone_from(BOOST_MOVE_BASE(Base, src), cloner, disposer); }
  1124. static treap &container_from_end_iterator(iterator end_iterator)
  1125. { return static_cast<treap &>(Base::container_from_end_iterator(end_iterator)); }
  1126. static const treap &container_from_end_iterator(const_iterator end_iterator)
  1127. { return static_cast<const treap &>(Base::container_from_end_iterator(end_iterator)); }
  1128. static treap &container_from_iterator(iterator it)
  1129. { return static_cast<treap &>(Base::container_from_iterator(it)); }
  1130. static const treap &container_from_iterator(const_iterator it)
  1131. { return static_cast<const treap &>(Base::container_from_iterator(it)); }
  1132. };
  1133. #endif
  1134. } //namespace intrusive
  1135. } //namespace boost
  1136. #include <boost/intrusive/detail/config_end.hpp>
  1137. #endif //BOOST_INTRUSIVE_TREAP_HPP