flat_set.hpp 56 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359
  1. //////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2005-2013. Distributed under the Boost
  4. // Software License, Version 1.0. (See accompanying file
  5. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. // See http://www.boost.org/libs/container for documentation.
  8. //
  9. //////////////////////////////////////////////////////////////////////////////
  10. #ifndef BOOST_CONTAINER_FLAT_SET_HPP
  11. #define BOOST_CONTAINER_FLAT_SET_HPP
  12. #ifndef BOOST_CONFIG_HPP
  13. # include <boost/config.hpp>
  14. #endif
  15. #if defined(BOOST_HAS_PRAGMA_ONCE)
  16. # pragma once
  17. #endif
  18. #include <boost/container/detail/config_begin.hpp>
  19. #include <boost/container/detail/workaround.hpp>
  20. // container
  21. #include <boost/container/allocator_traits.hpp>
  22. #include <boost/container/container_fwd.hpp>
  23. #include <boost/container/new_allocator.hpp> //new_allocator
  24. // container/detail
  25. #include <boost/container/detail/flat_tree.hpp>
  26. #include <boost/container/detail/mpl.hpp>
  27. // move
  28. #include <boost/move/traits.hpp>
  29. #include <boost/move/utility_core.hpp>
  30. // move/detail
  31. #if defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  32. #include <boost/move/detail/fwd_macros.hpp>
  33. #endif
  34. #include <boost/move/detail/move_helpers.hpp>
  35. // intrusive/detail
  36. #include <boost/intrusive/detail/minimal_pair_header.hpp> //pair
  37. #include <boost/intrusive/detail/minimal_less_equal_header.hpp>//less, equal
  38. // std
  39. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  40. #include <initializer_list>
  41. #endif
  42. namespace boost {
  43. namespace container {
  44. //! flat_set is a Sorted Associative Container that stores objects of type Key.
  45. //! It is also a Unique Associative Container, meaning that no two elements are the same.
  46. //!
  47. //! flat_set is similar to std::set but it's implemented like an ordered vector.
  48. //! This means that inserting a new element into a flat_set invalidates
  49. //! previous iterators and references
  50. //!
  51. //! Erasing an element of a flat_set invalidates iterators and references
  52. //! pointing to elements that come after (their keys are bigger) the erased element.
  53. //!
  54. //! This container provides random-access iterators.
  55. //!
  56. //! \tparam Key is the type to be inserted in the set, which is also the key_type
  57. //! \tparam Compare is the comparison functor used to order keys
  58. //! \tparam Allocator is the allocator to be used to allocate memory for this container
  59. #ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  60. template <class Key, class Compare = std::less<Key>, class Allocator = new_allocator<Key> >
  61. #else
  62. template <class Key, class Compare, class Allocator>
  63. #endif
  64. class flat_set
  65. ///@cond
  66. : public container_detail::flat_tree<Key, Key, container_detail::identity<Key>, Compare, Allocator>
  67. ///@endcond
  68. {
  69. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  70. private:
  71. BOOST_COPYABLE_AND_MOVABLE(flat_set)
  72. typedef container_detail::flat_tree<Key, Key, container_detail::identity<Key>, Compare, Allocator> base_t;
  73. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  74. public:
  75. //////////////////////////////////////////////
  76. //
  77. // types
  78. //
  79. //////////////////////////////////////////////
  80. typedef Key key_type;
  81. typedef Key value_type;
  82. typedef Compare key_compare;
  83. typedef Compare value_compare;
  84. typedef ::boost::container::allocator_traits<Allocator> allocator_traits_type;
  85. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  86. typedef typename ::boost::container::allocator_traits<Allocator>::const_pointer const_pointer;
  87. typedef typename ::boost::container::allocator_traits<Allocator>::reference reference;
  88. typedef typename ::boost::container::allocator_traits<Allocator>::const_reference const_reference;
  89. typedef typename ::boost::container::allocator_traits<Allocator>::size_type size_type;
  90. typedef typename ::boost::container::allocator_traits<Allocator>::difference_type difference_type;
  91. typedef Allocator allocator_type;
  92. typedef typename BOOST_CONTAINER_IMPDEF(base_t::stored_allocator_type) stored_allocator_type;
  93. typedef typename BOOST_CONTAINER_IMPDEF(base_t::iterator) iterator;
  94. typedef typename BOOST_CONTAINER_IMPDEF(base_t::const_iterator) const_iterator;
  95. typedef typename BOOST_CONTAINER_IMPDEF(base_t::reverse_iterator) reverse_iterator;
  96. typedef typename BOOST_CONTAINER_IMPDEF(base_t::const_reverse_iterator) const_reverse_iterator;
  97. public:
  98. //////////////////////////////////////////////
  99. //
  100. // construct/copy/destroy
  101. //
  102. //////////////////////////////////////////////
  103. //! <b>Effects</b>: Default constructs an empty container.
  104. //!
  105. //! <b>Complexity</b>: Constant.
  106. explicit flat_set()
  107. : base_t()
  108. {}
  109. //! <b>Effects</b>: Constructs an empty container using the specified
  110. //! comparison object and allocator.
  111. //!
  112. //! <b>Complexity</b>: Constant.
  113. explicit flat_set(const Compare& comp,
  114. const allocator_type& a = allocator_type())
  115. : base_t(comp, a)
  116. {}
  117. //! <b>Effects</b>: Constructs an empty container using the specified allocator.
  118. //!
  119. //! <b>Complexity</b>: Constant.
  120. explicit flat_set(const allocator_type& a)
  121. : base_t(a)
  122. {}
  123. //! <b>Effects</b>: Constructs an empty container using the specified comparison object and
  124. //! allocator, and inserts elements from the range [first ,last ).
  125. //!
  126. //! <b>Complexity</b>: Linear in N if the range [first ,last ) is already sorted using
  127. //! comp and otherwise N logN, where N is last - first.
  128. template <class InputIterator>
  129. flat_set(InputIterator first, InputIterator last,
  130. const Compare& comp = Compare(),
  131. const allocator_type& a = allocator_type())
  132. : base_t(true, first, last, comp, a)
  133. {}
  134. //! <b>Effects</b>: Constructs an empty container using the specified
  135. //! allocator, and inserts elements from the range [first ,last ).
  136. //!
  137. //! <b>Complexity</b>: Linear in N if the range [first ,last ) is already sorted using
  138. //! comp and otherwise N logN, where N is last - first.
  139. template <class InputIterator>
  140. flat_set(InputIterator first, InputIterator last, const allocator_type& a)
  141. : base_t(true, first, last, Compare(), a)
  142. {}
  143. //! <b>Effects</b>: Constructs an empty container using the specified comparison object and
  144. //! allocator, and inserts elements from the ordered unique range [first ,last). This function
  145. //! is more efficient than the normal range creation for ordered ranges.
  146. //!
  147. //! <b>Requires</b>: [first ,last) must be ordered according to the predicate and must be
  148. //! unique values.
  149. //!
  150. //! <b>Complexity</b>: Linear in N.
  151. //!
  152. //! <b>Note</b>: Non-standard extension.
  153. template <class InputIterator>
  154. flat_set(ordered_unique_range_t, InputIterator first, InputIterator last,
  155. const Compare& comp = Compare(),
  156. const allocator_type& a = allocator_type())
  157. : base_t(ordered_range, first, last, comp, a)
  158. {}
  159. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  160. //! <b>Effects</b>: Constructs an empty container using the specified comparison object and
  161. //! allocator, and inserts elements from the range [il.begin(), il.end()).
  162. //!
  163. //! <b>Complexity</b>: Linear in N if the range [il.begin(), il.end()) is already sorted using
  164. //! comp and otherwise N logN, where N is il.begin() - il.end().
  165. flat_set(std::initializer_list<value_type> il, const Compare& comp = Compare(),
  166. const allocator_type& a = allocator_type())
  167. : base_t(true, il.begin(), il.end(), comp, a)
  168. {}
  169. //! <b>Effects</b>: Constructs an empty container using the specified
  170. //! allocator, and inserts elements from the range [il.begin(), il.end()).
  171. //!
  172. //! <b>Complexity</b>: Linear in N if the range [il.begin(), il.end()) is already sorted using
  173. //! comp and otherwise N logN, where N is il.begin() - il.end().
  174. flat_set(std::initializer_list<value_type> il, const allocator_type& a)
  175. : base_t(true, il.begin(), il.end(), Compare(), a)
  176. {}
  177. //! <b>Effects</b>: Constructs an empty container using the specified comparison object and
  178. //! allocator, and inserts elements from the ordered unique range [il.begin(), il.end()). This function
  179. //! is more efficient than the normal range creation for ordered ranges.
  180. //!
  181. //! <b>Requires</b>: [il.begin(), il.end()) must be ordered according to the predicate and must be
  182. //! unique values.
  183. //!
  184. //! <b>Complexity</b>: Linear in N.
  185. //!
  186. //! <b>Note</b>: Non-standard extension.
  187. flat_set(ordered_unique_range_t, std::initializer_list<value_type> il,
  188. const Compare& comp = Compare(), const allocator_type& a = allocator_type())
  189. : base_t(ordered_range, il.begin(), il.end(), comp, a)
  190. {}
  191. #endif
  192. //! <b>Effects</b>: Copy constructs the container.
  193. //!
  194. //! <b>Complexity</b>: Linear in x.size().
  195. flat_set(const flat_set& x)
  196. : base_t(static_cast<const base_t&>(x))
  197. {}
  198. //! <b>Effects</b>: Move constructs thecontainer. Constructs *this using x's resources.
  199. //!
  200. //! <b>Complexity</b>: Constant.
  201. //!
  202. //! <b>Postcondition</b>: x is emptied.
  203. flat_set(BOOST_RV_REF(flat_set) x)
  204. : base_t(BOOST_MOVE_BASE(base_t, x))
  205. {}
  206. //! <b>Effects</b>: Copy constructs a container using the specified allocator.
  207. //!
  208. //! <b>Complexity</b>: Linear in x.size().
  209. flat_set(const flat_set& x, const allocator_type &a)
  210. : base_t(static_cast<const base_t&>(x), a)
  211. {}
  212. //! <b>Effects</b>: Move constructs a container using the specified allocator.
  213. //! Constructs *this using x's resources.
  214. //!
  215. //! <b>Complexity</b>: Constant if a == x.get_allocator(), linear otherwise
  216. flat_set(BOOST_RV_REF(flat_set) x, const allocator_type &a)
  217. : base_t(BOOST_MOVE_BASE(base_t, x), a)
  218. {}
  219. //! <b>Effects</b>: Makes *this a copy of x.
  220. //!
  221. //! <b>Complexity</b>: Linear in x.size().
  222. flat_set& operator=(BOOST_COPY_ASSIGN_REF(flat_set) x)
  223. { return static_cast<flat_set&>(this->base_t::operator=(static_cast<const base_t&>(x))); }
  224. //! <b>Throws</b>: If allocator_traits_type::propagate_on_container_move_assignment
  225. //! is false and (allocation throws or value_type's move constructor throws)
  226. //!
  227. //! <b>Complexity</b>: Constant if allocator_traits_type::
  228. //! propagate_on_container_move_assignment is true or
  229. //! this->get>allocator() == x.get_allocator(). Linear otherwise.
  230. flat_set& operator=(BOOST_RV_REF(flat_set) x)
  231. BOOST_NOEXCEPT_IF( allocator_traits_type::is_always_equal::value
  232. && boost::container::container_detail::is_nothrow_move_assignable<Compare>::value )
  233. { return static_cast<flat_set&>(this->base_t::operator=(BOOST_MOVE_BASE(base_t, x))); }
  234. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  235. //! <b>Effects</b>: Copy all elements from il to *this.
  236. //!
  237. //! <b>Complexity</b>: Linear in il.size().
  238. flat_set& operator=(std::initializer_list<value_type> il)
  239. {
  240. this->clear();
  241. this->insert(il.begin(), il.end());
  242. return *this;
  243. }
  244. #endif
  245. #ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  246. //! <b>Effects</b>: Returns a copy of the allocator that
  247. //! was passed to the object's constructor.
  248. //!
  249. //! <b>Complexity</b>: Constant.
  250. allocator_type get_allocator() const BOOST_NOEXCEPT_OR_NOTHROW;
  251. //! <b>Effects</b>: Returns a reference to the internal allocator.
  252. //!
  253. //! <b>Throws</b>: Nothing
  254. //!
  255. //! <b>Complexity</b>: Constant.
  256. //!
  257. //! <b>Note</b>: Non-standard extension.
  258. stored_allocator_type &get_stored_allocator() BOOST_NOEXCEPT_OR_NOTHROW;
  259. //! <b>Effects</b>: Returns a reference to the internal allocator.
  260. //!
  261. //! <b>Throws</b>: Nothing
  262. //!
  263. //! <b>Complexity</b>: Constant.
  264. //!
  265. //! <b>Note</b>: Non-standard extension.
  266. const stored_allocator_type &get_stored_allocator() const BOOST_NOEXCEPT_OR_NOTHROW;
  267. //! <b>Effects</b>: Returns an iterator to the first element contained in the container.
  268. //!
  269. //! <b>Throws</b>: Nothing.
  270. //!
  271. //! <b>Complexity</b>: Constant.
  272. iterator begin() BOOST_NOEXCEPT_OR_NOTHROW;
  273. //! <b>Effects</b>: Returns a const_iterator to the first element contained in the container.
  274. //!
  275. //! <b>Throws</b>: Nothing.
  276. //!
  277. //! <b>Complexity</b>: Constant.
  278. const_iterator begin() const BOOST_NOEXCEPT_OR_NOTHROW;
  279. //! <b>Effects</b>: Returns an iterator to the end of the container.
  280. //!
  281. //! <b>Throws</b>: Nothing.
  282. //!
  283. //! <b>Complexity</b>: Constant.
  284. iterator end() BOOST_NOEXCEPT_OR_NOTHROW;
  285. //! <b>Effects</b>: Returns a const_iterator to the end of the container.
  286. //!
  287. //! <b>Throws</b>: Nothing.
  288. //!
  289. //! <b>Complexity</b>: Constant.
  290. const_iterator end() const BOOST_NOEXCEPT_OR_NOTHROW;
  291. //! <b>Effects</b>: Returns a reverse_iterator pointing to the beginning
  292. //! of the reversed container.
  293. //!
  294. //! <b>Throws</b>: Nothing.
  295. //!
  296. //! <b>Complexity</b>: Constant.
  297. reverse_iterator rbegin() BOOST_NOEXCEPT_OR_NOTHROW;
  298. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the beginning
  299. //! of the reversed container.
  300. //!
  301. //! <b>Throws</b>: Nothing.
  302. //!
  303. //! <b>Complexity</b>: Constant.
  304. const_reverse_iterator rbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  305. //! <b>Effects</b>: Returns a reverse_iterator pointing to the end
  306. //! of the reversed container.
  307. //!
  308. //! <b>Throws</b>: Nothing.
  309. //!
  310. //! <b>Complexity</b>: Constant.
  311. reverse_iterator rend() BOOST_NOEXCEPT_OR_NOTHROW;
  312. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the end
  313. //! of the reversed container.
  314. //!
  315. //! <b>Throws</b>: Nothing.
  316. //!
  317. //! <b>Complexity</b>: Constant.
  318. const_reverse_iterator rend() const BOOST_NOEXCEPT_OR_NOTHROW;
  319. //! <b>Effects</b>: Returns a const_iterator to the first element contained in the container.
  320. //!
  321. //! <b>Throws</b>: Nothing.
  322. //!
  323. //! <b>Complexity</b>: Constant.
  324. const_iterator cbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  325. //! <b>Effects</b>: Returns a const_iterator to the end of the container.
  326. //!
  327. //! <b>Throws</b>: Nothing.
  328. //!
  329. //! <b>Complexity</b>: Constant.
  330. const_iterator cend() const BOOST_NOEXCEPT_OR_NOTHROW;
  331. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the beginning
  332. //! of the reversed container.
  333. //!
  334. //! <b>Throws</b>: Nothing.
  335. //!
  336. //! <b>Complexity</b>: Constant.
  337. const_reverse_iterator crbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  338. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the end
  339. //! of the reversed container.
  340. //!
  341. //! <b>Throws</b>: Nothing.
  342. //!
  343. //! <b>Complexity</b>: Constant.
  344. const_reverse_iterator crend() const BOOST_NOEXCEPT_OR_NOTHROW;
  345. //! <b>Effects</b>: Returns true if the container contains no elements.
  346. //!
  347. //! <b>Throws</b>: Nothing.
  348. //!
  349. //! <b>Complexity</b>: Constant.
  350. bool empty() const BOOST_NOEXCEPT_OR_NOTHROW;
  351. //! <b>Effects</b>: Returns the number of the elements contained in the container.
  352. //!
  353. //! <b>Throws</b>: Nothing.
  354. //!
  355. //! <b>Complexity</b>: Constant.
  356. size_type size() const BOOST_NOEXCEPT_OR_NOTHROW;
  357. //! <b>Effects</b>: Returns the largest possible size of the container.
  358. //!
  359. //! <b>Throws</b>: Nothing.
  360. //!
  361. //! <b>Complexity</b>: Constant.
  362. size_type max_size() const BOOST_NOEXCEPT_OR_NOTHROW;
  363. //! <b>Effects</b>: Number of elements for which memory has been allocated.
  364. //! capacity() is always greater than or equal to size().
  365. //!
  366. //! <b>Throws</b>: Nothing.
  367. //!
  368. //! <b>Complexity</b>: Constant.
  369. size_type capacity() const BOOST_NOEXCEPT_OR_NOTHROW;
  370. //! <b>Effects</b>: If n is less than or equal to capacity(), this call has no
  371. //! effect. Otherwise, it is a request for allocation of additional memory.
  372. //! If the request is successful, then capacity() is greater than or equal to
  373. //! n; otherwise, capacity() is unchanged. In either case, size() is unchanged.
  374. //!
  375. //! <b>Throws</b>: If memory allocation allocation throws or Key's copy constructor throws.
  376. //!
  377. //! <b>Note</b>: If capacity() is less than "cnt", iterators and references to
  378. //! to values might be invalidated.
  379. void reserve(size_type cnt);
  380. //! <b>Effects</b>: Tries to deallocate the excess of memory created
  381. // with previous allocations. The size of the vector is unchanged
  382. //!
  383. //! <b>Throws</b>: If memory allocation throws, or Key's copy constructor throws.
  384. //!
  385. //! <b>Complexity</b>: Linear to size().
  386. void shrink_to_fit();
  387. #endif // #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  388. //////////////////////////////////////////////
  389. //
  390. // modifiers
  391. //
  392. //////////////////////////////////////////////
  393. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  394. //! <b>Effects</b>: Inserts an object x of type Key constructed with
  395. //! std::forward<Args>(args)... if and only if there is no element in the container
  396. //! with key equivalent to the key of x.
  397. //!
  398. //! <b>Returns</b>: The bool component of the returned pair is true if and only
  399. //! if the insertion takes place, and the iterator component of the pair
  400. //! points to the element with key equivalent to the key of x.
  401. //!
  402. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  403. //! to the elements with bigger keys than x.
  404. //!
  405. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  406. template <class... Args>
  407. std::pair<iterator,bool> emplace(BOOST_FWD_REF(Args)... args)
  408. { return this->base_t::emplace_unique(boost::forward<Args>(args)...); }
  409. //! <b>Effects</b>: Inserts an object of type Key constructed with
  410. //! std::forward<Args>(args)... in the container if and only if there is
  411. //! no element in the container with key equivalent to the key of x.
  412. //! p is a hint pointing to where the insert should start to search.
  413. //!
  414. //! <b>Returns</b>: An iterator pointing to the element with key equivalent
  415. //! to the key of x.
  416. //!
  417. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  418. //! right before p) plus insertion linear to the elements with bigger keys than x.
  419. //!
  420. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  421. template <class... Args>
  422. iterator emplace_hint(const_iterator p, BOOST_FWD_REF(Args)... args)
  423. { return this->base_t::emplace_hint_unique(p, boost::forward<Args>(args)...); }
  424. #else // !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  425. #define BOOST_CONTAINER_FLAT_SET_EMPLACE_CODE(N) \
  426. BOOST_MOVE_TMPL_LT##N BOOST_MOVE_CLASS##N BOOST_MOVE_GT##N \
  427. std::pair<iterator,bool> emplace(BOOST_MOVE_UREF##N)\
  428. { return this->base_t::emplace_unique(BOOST_MOVE_FWD##N); }\
  429. \
  430. BOOST_MOVE_TMPL_LT##N BOOST_MOVE_CLASS##N BOOST_MOVE_GT##N \
  431. iterator emplace_hint(const_iterator hint BOOST_MOVE_I##N BOOST_MOVE_UREF##N)\
  432. { return this->base_t::emplace_hint_unique(hint BOOST_MOVE_I##N BOOST_MOVE_FWD##N); }\
  433. //
  434. BOOST_MOVE_ITERATE_0TO9(BOOST_CONTAINER_FLAT_SET_EMPLACE_CODE)
  435. #undef BOOST_CONTAINER_FLAT_SET_EMPLACE_CODE
  436. #endif // !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  437. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  438. //! <b>Effects</b>: Inserts x if and only if there is no element in the container
  439. //! with key equivalent to the key of x.
  440. //!
  441. //! <b>Returns</b>: The bool component of the returned pair is true if and only
  442. //! if the insertion takes place, and the iterator component of the pair
  443. //! points to the element with key equivalent to the key of x.
  444. //!
  445. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  446. //! to the elements with bigger keys than x.
  447. //!
  448. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  449. std::pair<iterator, bool> insert(const value_type &x);
  450. //! <b>Effects</b>: Inserts a new value_type move constructed from the pair if and
  451. //! only if there is no element in the container with key equivalent to the key of x.
  452. //!
  453. //! <b>Returns</b>: The bool component of the returned pair is true if and only
  454. //! if the insertion takes place, and the iterator component of the pair
  455. //! points to the element with key equivalent to the key of x.
  456. //!
  457. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  458. //! to the elements with bigger keys than x.
  459. //!
  460. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  461. std::pair<iterator, bool> insert(value_type &&x);
  462. #else
  463. private:
  464. typedef std::pair<iterator, bool> insert_return_pair;
  465. public:
  466. BOOST_MOVE_CONVERSION_AWARE_CATCH(insert, value_type, insert_return_pair, this->priv_insert)
  467. #endif
  468. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  469. //! <b>Effects</b>: Inserts a copy of x in the container if and only if there is
  470. //! no element in the container with key equivalent to the key of x.
  471. //! p is a hint pointing to where the insert should start to search.
  472. //!
  473. //! <b>Returns</b>: An iterator pointing to the element with key equivalent
  474. //! to the key of x.
  475. //!
  476. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  477. //! right before p) plus insertion linear to the elements with bigger keys than x.
  478. //!
  479. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  480. iterator insert(const_iterator p, const value_type &x);
  481. //! <b>Effects</b>: Inserts an element move constructed from x in the container.
  482. //! p is a hint pointing to where the insert should start to search.
  483. //!
  484. //! <b>Returns</b>: An iterator pointing to the element with key equivalent to the key of x.
  485. //!
  486. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  487. //! right before p) plus insertion linear to the elements with bigger keys than x.
  488. //!
  489. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  490. iterator insert(const_iterator p, value_type &&x);
  491. #else
  492. BOOST_MOVE_CONVERSION_AWARE_CATCH_1ARG(insert, value_type, iterator, this->priv_insert, const_iterator, const_iterator)
  493. #endif
  494. //! <b>Requires</b>: first, last are not iterators into *this.
  495. //!
  496. //! <b>Effects</b>: inserts each element from the range [first,last) if and only
  497. //! if there is no element with key equivalent to the key of that element.
  498. //!
  499. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from first to last)
  500. //! search time plus N*size() insertion time.
  501. //!
  502. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  503. template <class InputIterator>
  504. void insert(InputIterator first, InputIterator last)
  505. { this->base_t::insert_unique(first, last); }
  506. //! <b>Requires</b>: first, last are not iterators into *this and
  507. //! must be ordered according to the predicate and must be
  508. //! unique values.
  509. //!
  510. //! <b>Effects</b>: inserts each element from the range [first,last) .This function
  511. //! is more efficient than the normal range creation for ordered ranges.
  512. //!
  513. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from first to last)
  514. //! search time plus N*size() insertion time.
  515. //!
  516. //! <b>Note</b>: Non-standard extension. If an element is inserted it might invalidate elements.
  517. template <class InputIterator>
  518. void insert(ordered_unique_range_t, InputIterator first, InputIterator last)
  519. { this->base_t::insert_unique(ordered_unique_range, first, last); }
  520. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  521. //! <b>Effects</b>: inserts each element from the range [il.begin(), il.end()) if and only
  522. //! if there is no element with key equivalent to the key of that element.
  523. //!
  524. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from il.begin() to il.end())
  525. //! search time plus N*size() insertion time.
  526. //!
  527. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  528. void insert(std::initializer_list<value_type> il)
  529. { this->base_t::insert_unique(il.begin(), il.end()); }
  530. //! <b>Requires</b>: Range [il.begin(), il.end()) must be ordered according to the predicate
  531. //! and must be unique values.
  532. //!
  533. //! <b>Effects</b>: inserts each element from the range [il.begin(), il.end()) .This function
  534. //! is more efficient than the normal range creation for ordered ranges.
  535. //!
  536. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from il.begin() to il.end())
  537. //! search time plus N*size() insertion time.
  538. //!
  539. //! <b>Note</b>: Non-standard extension. If an element is inserted it might invalidate elements.
  540. void insert(ordered_unique_range_t, std::initializer_list<value_type> il)
  541. { this->base_t::insert_unique(ordered_unique_range, il.begin(), il.end()); }
  542. #endif
  543. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  544. //! <b>Effects</b>: Erases the element pointed to by p.
  545. //!
  546. //! <b>Returns</b>: Returns an iterator pointing to the element immediately
  547. //! following q prior to the element being erased. If no such element exists,
  548. //! returns end().
  549. //!
  550. //! <b>Complexity</b>: Linear to the elements with keys bigger than p
  551. //!
  552. //! <b>Note</b>: Invalidates elements with keys
  553. //! not less than the erased element.
  554. iterator erase(const_iterator p);
  555. //! <b>Effects</b>: Erases all elements in the container with key equivalent to x.
  556. //!
  557. //! <b>Returns</b>: Returns the number of erased elements.
  558. //!
  559. //! <b>Complexity</b>: Logarithmic search time plus erasure time
  560. //! linear to the elements with bigger keys.
  561. size_type erase(const key_type& x);
  562. //! <b>Effects</b>: Erases all the elements in the range [first, last).
  563. //!
  564. //! <b>Returns</b>: Returns last.
  565. //!
  566. //! <b>Complexity</b>: size()*N where N is the distance from first to last.
  567. //!
  568. //! <b>Complexity</b>: Logarithmic search time plus erasure time
  569. //! linear to the elements with bigger keys.
  570. iterator erase(const_iterator first, const_iterator last);
  571. //! <b>Effects</b>: Swaps the contents of *this and x.
  572. //!
  573. //! <b>Throws</b>: Nothing.
  574. //!
  575. //! <b>Complexity</b>: Constant.
  576. void swap(flat_set& x)
  577. BOOST_NOEXCEPT_IF( allocator_traits_type::is_always_equal::value
  578. && boost::container::container_detail::is_nothrow_swappable<Compare>::value );
  579. //! <b>Effects</b>: erase(a.begin(),a.end()).
  580. //!
  581. //! <b>Postcondition</b>: size() == 0.
  582. //!
  583. //! <b>Complexity</b>: linear in size().
  584. void clear() BOOST_NOEXCEPT_OR_NOTHROW;
  585. //! <b>Effects</b>: Returns the comparison object out
  586. //! of which a was constructed.
  587. //!
  588. //! <b>Complexity</b>: Constant.
  589. key_compare key_comp() const;
  590. //! <b>Effects</b>: Returns an object of value_compare constructed out
  591. //! of the comparison object.
  592. //!
  593. //! <b>Complexity</b>: Constant.
  594. value_compare value_comp() const;
  595. //! <b>Returns</b>: An iterator pointing to an element with the key
  596. //! equivalent to x, or end() if such an element is not found.
  597. //!
  598. //! <b>Complexity</b>: Logarithmic.
  599. iterator find(const key_type& x);
  600. //! <b>Returns</b>: A const_iterator pointing to an element with the key
  601. //! equivalent to x, or end() if such an element is not found.
  602. //!
  603. //! <b>Complexity</b>: Logarithmic.
  604. const_iterator find(const key_type& x) const;
  605. //! <b>Requires</b>: size() >= n.
  606. //!
  607. //! <b>Effects</b>: Returns an iterator to the nth element
  608. //! from the beginning of the container. Returns end()
  609. //! if n == size().
  610. //!
  611. //! <b>Throws</b>: Nothing.
  612. //!
  613. //! <b>Complexity</b>: Constant.
  614. //!
  615. //! <b>Note</b>: Non-standard extension
  616. iterator nth(size_type n) BOOST_NOEXCEPT_OR_NOTHROW;
  617. //! <b>Requires</b>: size() >= n.
  618. //!
  619. //! <b>Effects</b>: Returns a const_iterator to the nth element
  620. //! from the beginning of the container. Returns end()
  621. //! if n == size().
  622. //!
  623. //! <b>Throws</b>: Nothing.
  624. //!
  625. //! <b>Complexity</b>: Constant.
  626. //!
  627. //! <b>Note</b>: Non-standard extension
  628. const_iterator nth(size_type n) const BOOST_NOEXCEPT_OR_NOTHROW;
  629. //! <b>Requires</b>: size() >= n.
  630. //!
  631. //! <b>Effects</b>: Returns an iterator to the nth element
  632. //! from the beginning of the container. Returns end()
  633. //! if n == size().
  634. //!
  635. //! <b>Throws</b>: Nothing.
  636. //!
  637. //! <b>Complexity</b>: Constant.
  638. //!
  639. //! <b>Note</b>: Non-standard extension
  640. size_type index_of(iterator p) BOOST_NOEXCEPT_OR_NOTHROW;
  641. //! <b>Requires</b>: begin() <= p <= end().
  642. //!
  643. //! <b>Effects</b>: Returns the index of the element pointed by p
  644. //! and size() if p == end().
  645. //!
  646. //! <b>Throws</b>: Nothing.
  647. //!
  648. //! <b>Complexity</b>: Constant.
  649. //!
  650. //! <b>Note</b>: Non-standard extension
  651. size_type index_of(const_iterator p) const BOOST_NOEXCEPT_OR_NOTHROW;
  652. #endif // #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  653. //! <b>Returns</b>: The number of elements with key equivalent to x.
  654. //!
  655. //! <b>Complexity</b>: log(size())+count(k)
  656. size_type count(const key_type& x) const
  657. { return static_cast<size_type>(this->base_t::find(x) != this->base_t::cend()); }
  658. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  659. //! <b>Returns</b>: An iterator pointing to the first element with key not less
  660. //! than k, or a.end() if such an element is not found.
  661. //!
  662. //! <b>Complexity</b>: Logarithmic
  663. iterator lower_bound(const key_type& x);
  664. //! <b>Returns</b>: A const iterator pointing to the first element with key not
  665. //! less than k, or a.end() if such an element is not found.
  666. //!
  667. //! <b>Complexity</b>: Logarithmic
  668. const_iterator lower_bound(const key_type& x) const;
  669. //! <b>Returns</b>: An iterator pointing to the first element with key not less
  670. //! than x, or end() if such an element is not found.
  671. //!
  672. //! <b>Complexity</b>: Logarithmic
  673. iterator upper_bound(const key_type& x);
  674. //! <b>Returns</b>: A const iterator pointing to the first element with key not
  675. //! less than x, or end() if such an element is not found.
  676. //!
  677. //! <b>Complexity</b>: Logarithmic
  678. const_iterator upper_bound(const key_type& x) const;
  679. #endif // #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  680. //! <b>Effects</b>: Equivalent to std::make_pair(this->lower_bound(k), this->upper_bound(k)).
  681. //!
  682. //! <b>Complexity</b>: Logarithmic
  683. std::pair<const_iterator, const_iterator> equal_range(const key_type& x) const
  684. { return this->base_t::lower_bound_range(x); }
  685. //! <b>Effects</b>: Equivalent to std::make_pair(this->lower_bound(k), this->upper_bound(k)).
  686. //!
  687. //! <b>Complexity</b>: Logarithmic
  688. std::pair<iterator,iterator> equal_range(const key_type& x)
  689. { return this->base_t::lower_bound_range(x); }
  690. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  691. //! <b>Effects</b>: Returns true if x and y are equal
  692. //!
  693. //! <b>Complexity</b>: Linear to the number of elements in the container.
  694. friend bool operator==(const flat_set& x, const flat_set& y);
  695. //! <b>Effects</b>: Returns true if x and y are unequal
  696. //!
  697. //! <b>Complexity</b>: Linear to the number of elements in the container.
  698. friend bool operator!=(const flat_set& x, const flat_set& y);
  699. //! <b>Effects</b>: Returns true if x is less than y
  700. //!
  701. //! <b>Complexity</b>: Linear to the number of elements in the container.
  702. friend bool operator<(const flat_set& x, const flat_set& y);
  703. //! <b>Effects</b>: Returns true if x is greater than y
  704. //!
  705. //! <b>Complexity</b>: Linear to the number of elements in the container.
  706. friend bool operator>(const flat_set& x, const flat_set& y);
  707. //! <b>Effects</b>: Returns true if x is equal or less than y
  708. //!
  709. //! <b>Complexity</b>: Linear to the number of elements in the container.
  710. friend bool operator<=(const flat_set& x, const flat_set& y);
  711. //! <b>Effects</b>: Returns true if x is equal or greater than y
  712. //!
  713. //! <b>Complexity</b>: Linear to the number of elements in the container.
  714. friend bool operator>=(const flat_set& x, const flat_set& y);
  715. //! <b>Effects</b>: x.swap(y)
  716. //!
  717. //! <b>Complexity</b>: Constant.
  718. friend void swap(flat_set& x, flat_set& y);
  719. #endif //#ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  720. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  721. private:
  722. template<class KeyType>
  723. std::pair<iterator, bool> priv_insert(BOOST_FWD_REF(KeyType) x)
  724. { return this->base_t::insert_unique(::boost::forward<KeyType>(x)); }
  725. template<class KeyType>
  726. iterator priv_insert(const_iterator p, BOOST_FWD_REF(KeyType) x)
  727. { return this->base_t::insert_unique(p, ::boost::forward<KeyType>(x)); }
  728. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  729. };
  730. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  731. } //namespace container {
  732. //!has_trivial_destructor_after_move<> == true_type
  733. //!specialization for optimizations
  734. template <class Key, class Compare, class Allocator>
  735. struct has_trivial_destructor_after_move<boost::container::flat_set<Key, Compare, Allocator> >
  736. {
  737. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  738. static const bool value = ::boost::has_trivial_destructor_after_move<Allocator>::value &&
  739. ::boost::has_trivial_destructor_after_move<pointer>::value &&
  740. ::boost::has_trivial_destructor_after_move<Compare>::value;
  741. };
  742. namespace container {
  743. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  744. //! flat_multiset is a Sorted Associative Container that stores objects of type Key.
  745. //!
  746. //! flat_multiset can store multiple copies of the same key value.
  747. //!
  748. //! flat_multiset is similar to std::multiset but it's implemented like an ordered vector.
  749. //! This means that inserting a new element into a flat_multiset invalidates
  750. //! previous iterators and references
  751. //!
  752. //! Erasing an element invalidates iterators and references
  753. //! pointing to elements that come after (their keys are bigger) the erased element.
  754. //!
  755. //! This container provides random-access iterators.
  756. //!
  757. //! \tparam Key is the type to be inserted in the multiset, which is also the key_type
  758. //! \tparam Compare is the comparison functor used to order keys
  759. //! \tparam Allocator is the allocator to be used to allocate memory for this container
  760. #ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  761. template <class Key, class Compare = std::less<Key>, class Allocator = new_allocator<Key> >
  762. #else
  763. template <class Key, class Compare, class Allocator>
  764. #endif
  765. class flat_multiset
  766. ///@cond
  767. : public container_detail::flat_tree<Key, Key, container_detail::identity<Key>, Compare, Allocator>
  768. ///@endcond
  769. {
  770. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  771. private:
  772. BOOST_COPYABLE_AND_MOVABLE(flat_multiset)
  773. typedef container_detail::flat_tree<Key, Key, container_detail::identity<Key>, Compare, Allocator> base_t;
  774. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  775. public:
  776. //////////////////////////////////////////////
  777. //
  778. // types
  779. //
  780. //////////////////////////////////////////////
  781. typedef Key key_type;
  782. typedef Key value_type;
  783. typedef Compare key_compare;
  784. typedef Compare value_compare;
  785. typedef ::boost::container::allocator_traits<Allocator> allocator_traits_type;
  786. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  787. typedef typename ::boost::container::allocator_traits<Allocator>::const_pointer const_pointer;
  788. typedef typename ::boost::container::allocator_traits<Allocator>::reference reference;
  789. typedef typename ::boost::container::allocator_traits<Allocator>::const_reference const_reference;
  790. typedef typename ::boost::container::allocator_traits<Allocator>::size_type size_type;
  791. typedef typename ::boost::container::allocator_traits<Allocator>::difference_type difference_type;
  792. typedef Allocator allocator_type;
  793. typedef typename BOOST_CONTAINER_IMPDEF(base_t::stored_allocator_type) stored_allocator_type;
  794. typedef typename BOOST_CONTAINER_IMPDEF(base_t::iterator) iterator;
  795. typedef typename BOOST_CONTAINER_IMPDEF(base_t::const_iterator) const_iterator;
  796. typedef typename BOOST_CONTAINER_IMPDEF(base_t::reverse_iterator) reverse_iterator;
  797. typedef typename BOOST_CONTAINER_IMPDEF(base_t::const_reverse_iterator) const_reverse_iterator;
  798. //! @copydoc ::boost::container::flat_set::flat_set()
  799. explicit flat_multiset()
  800. : base_t()
  801. {}
  802. //! @copydoc ::boost::container::flat_set::flat_set(const Compare&, const allocator_type&)
  803. explicit flat_multiset(const Compare& comp,
  804. const allocator_type& a = allocator_type())
  805. : base_t(comp, a)
  806. {}
  807. //! @copydoc ::boost::container::flat_set::flat_set(const allocator_type&)
  808. explicit flat_multiset(const allocator_type& a)
  809. : base_t(a)
  810. {}
  811. //! @copydoc ::boost::container::flat_set::flat_set(InputIterator, InputIterator, const Compare& comp, const allocator_type&)
  812. template <class InputIterator>
  813. flat_multiset(InputIterator first, InputIterator last,
  814. const Compare& comp = Compare(),
  815. const allocator_type& a = allocator_type())
  816. : base_t(false, first, last, comp, a)
  817. {}
  818. //! @copydoc ::boost::container::flat_set::flat_set(InputIterator, InputIterator, const allocator_type&)
  819. template <class InputIterator>
  820. flat_multiset(InputIterator first, InputIterator last, const allocator_type& a)
  821. : base_t(false, first, last, Compare(), a)
  822. {}
  823. //! <b>Effects</b>: Constructs an empty flat_multiset using the specified comparison object and
  824. //! allocator, and inserts elements from the ordered range [first ,last ). This function
  825. //! is more efficient than the normal range creation for ordered ranges.
  826. //!
  827. //! <b>Requires</b>: [first ,last) must be ordered according to the predicate.
  828. //!
  829. //! <b>Complexity</b>: Linear in N.
  830. //!
  831. //! <b>Note</b>: Non-standard extension.
  832. template <class InputIterator>
  833. flat_multiset(ordered_range_t, InputIterator first, InputIterator last,
  834. const Compare& comp = Compare(),
  835. const allocator_type& a = allocator_type())
  836. : base_t(ordered_range, first, last, comp, a)
  837. {}
  838. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  839. //! @copydoc ::boost::container::flat_set::flat_set(std::initializer_list<value_type>, const Compare& comp, const allocator_type&)
  840. flat_multiset(std::initializer_list<value_type> il, const Compare& comp = Compare(),
  841. const allocator_type& a = allocator_type())
  842. : base_t(false, il.begin(), il.end(), comp, a)
  843. {}
  844. //! @copydoc ::boost::container::flat_set::flat_set(std::initializer_list<value_type>, const allocator_type&)
  845. flat_multiset(std::initializer_list<value_type> il, const allocator_type& a)
  846. : base_t(false, il.begin(), il.end(), Compare(), a)
  847. {}
  848. //! @copydoc ::boost::container::flat_set::flat_set(ordered_unique_range_t, std::initializer_list<value_type>, const Compare& comp, const allocator_type&)
  849. flat_multiset(ordered_unique_range_t, std::initializer_list<value_type> il,
  850. const Compare& comp = Compare(), const allocator_type& a = allocator_type())
  851. : base_t(ordered_range, il.begin(), il.end(), comp, a)
  852. {}
  853. #endif
  854. //! @copydoc ::boost::container::flat_set::flat_set(const flat_set &)
  855. flat_multiset(const flat_multiset& x)
  856. : base_t(static_cast<const base_t&>(x))
  857. {}
  858. //! @copydoc ::boost::container::flat_set::flat_set(flat_set &&)
  859. flat_multiset(BOOST_RV_REF(flat_multiset) x)
  860. : base_t(boost::move(static_cast<base_t&>(x)))
  861. {}
  862. //! @copydoc ::boost::container::flat_set::flat_set(const flat_set &, const allocator_type &)
  863. flat_multiset(const flat_multiset& x, const allocator_type &a)
  864. : base_t(static_cast<const base_t&>(x), a)
  865. {}
  866. //! @copydoc ::boost::container::flat_set::flat_set(flat_set &&, const allocator_type &)
  867. flat_multiset(BOOST_RV_REF(flat_multiset) x, const allocator_type &a)
  868. : base_t(BOOST_MOVE_BASE(base_t, x), a)
  869. {}
  870. //! @copydoc ::boost::container::flat_set::operator=(const flat_set &)
  871. flat_multiset& operator=(BOOST_COPY_ASSIGN_REF(flat_multiset) x)
  872. { return static_cast<flat_multiset&>(this->base_t::operator=(static_cast<const base_t&>(x))); }
  873. //! @copydoc ::boost::container::flat_set::operator=(flat_set &&)
  874. flat_multiset& operator=(BOOST_RV_REF(flat_multiset) x)
  875. BOOST_NOEXCEPT_IF( allocator_traits_type::is_always_equal::value
  876. && boost::container::container_detail::is_nothrow_move_assignable<Compare>::value )
  877. { return static_cast<flat_multiset&>(this->base_t::operator=(BOOST_MOVE_BASE(base_t, x))); }
  878. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  879. //! @copydoc ::boost::container::flat_set::operator=(std::initializer_list<value_type>)
  880. flat_multiset& operator=(std::initializer_list<value_type> il)
  881. {
  882. this->clear();
  883. this->insert(il.begin(), il.end());
  884. return *this;
  885. }
  886. #endif
  887. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  888. //! @copydoc ::boost::container::flat_set::get_allocator()
  889. allocator_type get_allocator() const BOOST_NOEXCEPT_OR_NOTHROW;
  890. //! @copydoc ::boost::container::flat_set::get_stored_allocator()
  891. stored_allocator_type &get_stored_allocator() BOOST_NOEXCEPT_OR_NOTHROW;
  892. //! @copydoc ::boost::container::flat_set::get_stored_allocator() const
  893. const stored_allocator_type &get_stored_allocator() const BOOST_NOEXCEPT_OR_NOTHROW;
  894. //! @copydoc ::boost::container::flat_set::begin()
  895. iterator begin() BOOST_NOEXCEPT_OR_NOTHROW;
  896. //! @copydoc ::boost::container::flat_set::begin() const
  897. const_iterator begin() const;
  898. //! @copydoc ::boost::container::flat_set::cbegin() const
  899. const_iterator cbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  900. //! @copydoc ::boost::container::flat_set::end()
  901. iterator end() BOOST_NOEXCEPT_OR_NOTHROW;
  902. //! @copydoc ::boost::container::flat_set::end() const
  903. const_iterator end() const BOOST_NOEXCEPT_OR_NOTHROW;
  904. //! @copydoc ::boost::container::flat_set::cend() const
  905. const_iterator cend() const BOOST_NOEXCEPT_OR_NOTHROW;
  906. //! @copydoc ::boost::container::flat_set::rbegin()
  907. reverse_iterator rbegin() BOOST_NOEXCEPT_OR_NOTHROW;
  908. //! @copydoc ::boost::container::flat_set::rbegin() const
  909. const_reverse_iterator rbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  910. //! @copydoc ::boost::container::flat_set::crbegin() const
  911. const_reverse_iterator crbegin() const BOOST_NOEXCEPT_OR_NOTHROW;
  912. //! @copydoc ::boost::container::flat_set::rend()
  913. reverse_iterator rend() BOOST_NOEXCEPT_OR_NOTHROW;
  914. //! @copydoc ::boost::container::flat_set::rend() const
  915. const_reverse_iterator rend() const BOOST_NOEXCEPT_OR_NOTHROW;
  916. //! @copydoc ::boost::container::flat_set::crend() const
  917. const_reverse_iterator crend() const BOOST_NOEXCEPT_OR_NOTHROW;
  918. //! @copydoc ::boost::container::flat_set::empty() const
  919. bool empty() const BOOST_NOEXCEPT_OR_NOTHROW;
  920. //! @copydoc ::boost::container::flat_set::size() const
  921. size_type size() const BOOST_NOEXCEPT_OR_NOTHROW;
  922. //! @copydoc ::boost::container::flat_set::max_size() const
  923. size_type max_size() const BOOST_NOEXCEPT_OR_NOTHROW;
  924. //! @copydoc ::boost::container::flat_set::capacity() const
  925. size_type capacity() const BOOST_NOEXCEPT_OR_NOTHROW;
  926. //! @copydoc ::boost::container::flat_set::reserve(size_type)
  927. void reserve(size_type cnt);
  928. //! @copydoc ::boost::container::flat_set::shrink_to_fit()
  929. void shrink_to_fit();
  930. #endif // #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  931. //////////////////////////////////////////////
  932. //
  933. // modifiers
  934. //
  935. //////////////////////////////////////////////
  936. #if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  937. //! <b>Effects</b>: Inserts an object of type Key constructed with
  938. //! std::forward<Args>(args)... and returns the iterator pointing to the
  939. //! newly inserted element.
  940. //!
  941. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  942. //! to the elements with bigger keys than x.
  943. //!
  944. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  945. template <class... Args>
  946. iterator emplace(BOOST_FWD_REF(Args)... args)
  947. { return this->base_t::emplace_equal(boost::forward<Args>(args)...); }
  948. //! <b>Effects</b>: Inserts an object of type Key constructed with
  949. //! std::forward<Args>(args)... in the container.
  950. //! p is a hint pointing to where the insert should start to search.
  951. //!
  952. //! <b>Returns</b>: An iterator pointing to the element with key equivalent
  953. //! to the key of x.
  954. //!
  955. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  956. //! right before p) plus insertion linear to the elements with bigger keys than x.
  957. //!
  958. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  959. template <class... Args>
  960. iterator emplace_hint(const_iterator p, BOOST_FWD_REF(Args)... args)
  961. { return this->base_t::emplace_hint_equal(p, boost::forward<Args>(args)...); }
  962. #else // !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  963. #define BOOST_CONTAINER_FLAT_MULTISET_EMPLACE_CODE(N) \
  964. BOOST_MOVE_TMPL_LT##N BOOST_MOVE_CLASS##N BOOST_MOVE_GT##N \
  965. iterator emplace(BOOST_MOVE_UREF##N)\
  966. { return this->base_t::emplace_equal(BOOST_MOVE_FWD##N); }\
  967. \
  968. BOOST_MOVE_TMPL_LT##N BOOST_MOVE_CLASS##N BOOST_MOVE_GT##N \
  969. iterator emplace_hint(const_iterator hint BOOST_MOVE_I##N BOOST_MOVE_UREF##N)\
  970. { return this->base_t::emplace_hint_equal(hint BOOST_MOVE_I##N BOOST_MOVE_FWD##N); }\
  971. //
  972. BOOST_MOVE_ITERATE_0TO9(BOOST_CONTAINER_FLAT_MULTISET_EMPLACE_CODE)
  973. #undef BOOST_CONTAINER_FLAT_MULTISET_EMPLACE_CODE
  974. #endif // !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
  975. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  976. //! <b>Effects</b>: Inserts x and returns the iterator pointing to the
  977. //! newly inserted element.
  978. //!
  979. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  980. //! to the elements with bigger keys than x.
  981. //!
  982. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  983. iterator insert(const value_type &x);
  984. //! <b>Effects</b>: Inserts a new value_type move constructed from x
  985. //! and returns the iterator pointing to the newly inserted element.
  986. //!
  987. //! <b>Complexity</b>: Logarithmic search time plus linear insertion
  988. //! to the elements with bigger keys than x.
  989. //!
  990. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  991. iterator insert(value_type &&x);
  992. #else
  993. BOOST_MOVE_CONVERSION_AWARE_CATCH(insert, value_type, iterator, this->priv_insert)
  994. #endif
  995. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  996. //! <b>Effects</b>: Inserts a copy of x in the container.
  997. //! p is a hint pointing to where the insert should start to search.
  998. //!
  999. //! <b>Returns</b>: An iterator pointing to the element with key equivalent
  1000. //! to the key of x.
  1001. //!
  1002. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  1003. //! right before p) plus insertion linear to the elements with bigger keys than x.
  1004. //!
  1005. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  1006. iterator insert(const_iterator p, const value_type &x);
  1007. //! <b>Effects</b>: Inserts a new value move constructed from x in the container.
  1008. //! p is a hint pointing to where the insert should start to search.
  1009. //!
  1010. //! <b>Returns</b>: An iterator pointing to the element with key equivalent
  1011. //! to the key of x.
  1012. //!
  1013. //! <b>Complexity</b>: Logarithmic search time (constant if x is inserted
  1014. //! right before p) plus insertion linear to the elements with bigger keys than x.
  1015. //!
  1016. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  1017. iterator insert(const_iterator p, value_type &&x);
  1018. #else
  1019. BOOST_MOVE_CONVERSION_AWARE_CATCH_1ARG(insert, value_type, iterator, this->priv_insert, const_iterator, const_iterator)
  1020. #endif
  1021. //! <b>Requires</b>: first, last are not iterators into *this.
  1022. //!
  1023. //! <b>Effects</b>: inserts each element from the range [first,last) .
  1024. //!
  1025. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from first to last)
  1026. //! search time plus N*size() insertion time.
  1027. //!
  1028. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  1029. template <class InputIterator>
  1030. void insert(InputIterator first, InputIterator last)
  1031. { this->base_t::insert_equal(first, last); }
  1032. //! <b>Requires</b>: first, last are not iterators into *this and
  1033. //! must be ordered according to the predicate.
  1034. //!
  1035. //! <b>Effects</b>: inserts each element from the range [first,last) .This function
  1036. //! is more efficient than the normal range creation for ordered ranges.
  1037. //!
  1038. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from first to last)
  1039. //! search time plus N*size() insertion time.
  1040. //!
  1041. //! <b>Note</b>: Non-standard extension. If an element is inserted it might invalidate elements.
  1042. template <class InputIterator>
  1043. void insert(ordered_range_t, InputIterator first, InputIterator last)
  1044. { this->base_t::insert_equal(ordered_range, first, last); }
  1045. #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
  1046. //! <b>Effects</b>: inserts each element from the range [il.begin(), il.end()).
  1047. //!
  1048. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from first to last)
  1049. //! search time plus N*size() insertion time.
  1050. //!
  1051. //! <b>Note</b>: If an element is inserted it might invalidate elements.
  1052. void insert(std::initializer_list<value_type> il)
  1053. { this->base_t::insert_equal(il.begin(), il.end()); }
  1054. //! <b>Requires</b>: Range [il.begin(), il.end()) must be ordered according to the predicate.
  1055. //!
  1056. //! <b>Effects</b>: inserts each element from the range [il.begin(), il.end()). This function
  1057. //! is more efficient than the normal range creation for ordered ranges.
  1058. //!
  1059. //! <b>Complexity</b>: At most N log(size()+N) (N is the distance from il.begin() to il.end())
  1060. //! search time plus N*size() insertion time.
  1061. //!
  1062. //! <b>Note</b>: Non-standard extension. If an element is inserted it might invalidate elements.
  1063. void insert(ordered_range_t, std::initializer_list<value_type> il)
  1064. { this->base_t::insert_equal(ordered_range, il.begin(), il.end()); }
  1065. #endif
  1066. #if defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1067. //! @copydoc ::boost::container::flat_set::erase(const_iterator)
  1068. iterator erase(const_iterator p);
  1069. //! @copydoc ::boost::container::flat_set::erase(const key_type&)
  1070. size_type erase(const key_type& x);
  1071. //! @copydoc ::boost::container::flat_set::erase(const_iterator,const_iterator)
  1072. iterator erase(const_iterator first, const_iterator last);
  1073. //! @copydoc ::boost::container::flat_set::swap
  1074. void swap(flat_multiset& x)
  1075. BOOST_NOEXCEPT_IF( allocator_traits_type::is_always_equal::value
  1076. && boost::container::container_detail::is_nothrow_swappable<Compare>::value );
  1077. //! @copydoc ::boost::container::flat_set::clear
  1078. void clear() BOOST_NOEXCEPT_OR_NOTHROW;
  1079. //! @copydoc ::boost::container::flat_set::key_comp
  1080. key_compare key_comp() const;
  1081. //! @copydoc ::boost::container::flat_set::value_comp
  1082. value_compare value_comp() const;
  1083. //! @copydoc ::boost::container::flat_set::find(const key_type& )
  1084. iterator find(const key_type& x);
  1085. //! @copydoc ::boost::container::flat_set::find(const key_type& ) const
  1086. const_iterator find(const key_type& x) const;
  1087. //! @copydoc ::boost::container::flat_set::nth(size_type)
  1088. iterator nth(size_type n) BOOST_NOEXCEPT_OR_NOTHROW;
  1089. //! @copydoc ::boost::container::flat_set::nth(size_type) const
  1090. const_iterator nth(size_type n) const BOOST_NOEXCEPT_OR_NOTHROW;
  1091. //! @copydoc ::boost::container::flat_set::index_of(iterator)
  1092. size_type index_of(iterator p) BOOST_NOEXCEPT_OR_NOTHROW;
  1093. //! @copydoc ::boost::container::flat_set::index_of(const_iterator) const
  1094. size_type index_of(const_iterator p) const BOOST_NOEXCEPT_OR_NOTHROW;
  1095. //! @copydoc ::boost::container::flat_set::count(const key_type& ) const
  1096. size_type count(const key_type& x) const;
  1097. //! @copydoc ::boost::container::flat_set::lower_bound(const key_type& )
  1098. iterator lower_bound(const key_type& x);
  1099. //! @copydoc ::boost::container::flat_set::lower_bound(const key_type& ) const
  1100. const_iterator lower_bound(const key_type& x) const;
  1101. //! @copydoc ::boost::container::flat_set::upper_bound(const key_type& )
  1102. iterator upper_bound(const key_type& x);
  1103. //! @copydoc ::boost::container::flat_set::upper_bound(const key_type& ) const
  1104. const_iterator upper_bound(const key_type& x) const;
  1105. //! @copydoc ::boost::container::flat_set::equal_range(const key_type& ) const
  1106. std::pair<const_iterator, const_iterator> equal_range(const key_type& x) const;
  1107. //! @copydoc ::boost::container::flat_set::equal_range(const key_type& )
  1108. std::pair<iterator,iterator> equal_range(const key_type& x);
  1109. //! <b>Effects</b>: Returns true if x and y are equal
  1110. //!
  1111. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1112. friend bool operator==(const flat_multiset& x, const flat_multiset& y);
  1113. //! <b>Effects</b>: Returns true if x and y are unequal
  1114. //!
  1115. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1116. friend bool operator!=(const flat_multiset& x, const flat_multiset& y);
  1117. //! <b>Effects</b>: Returns true if x is less than y
  1118. //!
  1119. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1120. friend bool operator<(const flat_multiset& x, const flat_multiset& y);
  1121. //! <b>Effects</b>: Returns true if x is greater than y
  1122. //!
  1123. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1124. friend bool operator>(const flat_multiset& x, const flat_multiset& y);
  1125. //! <b>Effects</b>: Returns true if x is equal or less than y
  1126. //!
  1127. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1128. friend bool operator<=(const flat_multiset& x, const flat_multiset& y);
  1129. //! <b>Effects</b>: Returns true if x is equal or greater than y
  1130. //!
  1131. //! <b>Complexity</b>: Linear to the number of elements in the container.
  1132. friend bool operator>=(const flat_multiset& x, const flat_multiset& y);
  1133. //! <b>Effects</b>: x.swap(y)
  1134. //!
  1135. //! <b>Complexity</b>: Constant.
  1136. friend void swap(flat_multiset& x, flat_multiset& y);
  1137. #endif //#ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  1138. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  1139. private:
  1140. template <class KeyType>
  1141. iterator priv_insert(BOOST_FWD_REF(KeyType) x)
  1142. { return this->base_t::insert_equal(::boost::forward<KeyType>(x)); }
  1143. template <class KeyType>
  1144. iterator priv_insert(const_iterator p, BOOST_FWD_REF(KeyType) x)
  1145. { return this->base_t::insert_equal(p, ::boost::forward<KeyType>(x)); }
  1146. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  1147. };
  1148. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  1149. } //namespace container {
  1150. //!has_trivial_destructor_after_move<> == true_type
  1151. //!specialization for optimizations
  1152. template <class Key, class Compare, class Allocator>
  1153. struct has_trivial_destructor_after_move<boost::container::flat_multiset<Key, Compare, Allocator> >
  1154. {
  1155. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  1156. static const bool value = ::boost::has_trivial_destructor_after_move<Allocator>::value &&
  1157. ::boost::has_trivial_destructor_after_move<pointer>::value &&
  1158. ::boost::has_trivial_destructor_after_move<Compare>::value;
  1159. };
  1160. namespace container {
  1161. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  1162. }}
  1163. #include <boost/container/detail/config_end.hpp>
  1164. #endif // BOOST_CONTAINER_FLAT_SET_HPP