string.hpp 108 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051
  1. //////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2005-2015. 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_STRING_HPP
  11. #define BOOST_CONTAINER_STRING_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. #include <boost/container/container_fwd.hpp>
  21. // container
  22. #include <boost/container/allocator_traits.hpp>
  23. #include <boost/container/new_allocator.hpp> //new_allocator
  24. #include <boost/container/throw_exception.hpp>
  25. // container/detail
  26. #include <boost/container/detail/alloc_helpers.hpp>
  27. #include <boost/container/detail/allocator_version_traits.hpp>
  28. #include <boost/container/detail/allocation_type.hpp>
  29. #include <boost/container/detail/iterator.hpp>
  30. #include <boost/container/detail/iterators.hpp>
  31. #include <boost/container/detail/min_max.hpp>
  32. #include <boost/container/detail/mpl.hpp>
  33. #include <boost/container/detail/next_capacity.hpp>
  34. #include <boost/container/detail/to_raw_pointer.hpp>
  35. #include <boost/container/detail/version_type.hpp>
  36. #include <boost/move/utility_core.hpp>
  37. #include <boost/move/adl_move_swap.hpp>
  38. #include <boost/static_assert.hpp>
  39. #include <boost/intrusive/pointer_traits.hpp>
  40. #include <boost/core/no_exceptions_support.hpp>
  41. #include <boost/container/detail/minimal_char_traits_header.hpp>
  42. #include <boost/functional/hash.hpp>
  43. #include <algorithm>
  44. #include <functional> //bind2nd, etc.
  45. #include <iosfwd>
  46. #include <istream>
  47. #include <ostream>
  48. #include <ios>
  49. #include <locale>
  50. #include <cstddef>
  51. #include <climits>
  52. #include <boost/container/detail/type_traits.hpp>
  53. #include <boost/move/traits.hpp>
  54. namespace boost {
  55. namespace container {
  56. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  57. namespace container_detail {
  58. // ------------------------------------------------------------
  59. // Class basic_string_base.
  60. // basic_string_base is a helper class that makes it it easier to write
  61. // an exception-safe version of basic_string. The constructor allocates,
  62. // but does not initialize, a block of memory. The destructor
  63. // deallocates, but does not destroy elements within, a block of
  64. // memory. The destructor assumes that the memory either is the internal buffer,
  65. // or else points to a block of memory that was allocated using string_base's
  66. // allocator and whose size is this->m_storage.
  67. template <class Allocator>
  68. class basic_string_base
  69. {
  70. basic_string_base & operator=(const basic_string_base &);
  71. basic_string_base(const basic_string_base &);
  72. typedef allocator_traits<Allocator> allocator_traits_type;
  73. public:
  74. typedef Allocator allocator_type;
  75. typedef allocator_type stored_allocator_type;
  76. typedef typename allocator_traits_type::pointer pointer;
  77. typedef typename allocator_traits_type::value_type value_type;
  78. typedef typename allocator_traits_type::size_type size_type;
  79. typedef ::boost::intrusive::pointer_traits<pointer> pointer_traits;
  80. basic_string_base()
  81. : members_()
  82. { init(); }
  83. explicit basic_string_base(const allocator_type& a)
  84. : members_(a)
  85. { init(); }
  86. explicit basic_string_base(BOOST_RV_REF(allocator_type) a)
  87. : members_(boost::move(a))
  88. { this->init(); }
  89. basic_string_base(const allocator_type& a, size_type n)
  90. : members_(a)
  91. {
  92. this->init();
  93. this->allocate_initial_block(n);
  94. }
  95. explicit basic_string_base(size_type n)
  96. : members_()
  97. {
  98. this->init();
  99. this->allocate_initial_block(n);
  100. }
  101. ~basic_string_base()
  102. {
  103. if(!this->is_short()){
  104. this->deallocate(this->priv_long_addr(), this->priv_long_storage());
  105. }
  106. }
  107. private:
  108. //This is the structure controlling a long string
  109. struct long_t
  110. {
  111. size_type is_short : 1;
  112. size_type length : (sizeof(size_type)*CHAR_BIT - 1);
  113. size_type storage;
  114. pointer start;
  115. long_t()
  116. {}
  117. long_t(const long_t &other)
  118. {
  119. this->is_short = false;
  120. length = other.length;
  121. storage = other.storage;
  122. start = other.start;
  123. }
  124. long_t &operator= (const long_t &other)
  125. {
  126. length = other.length;
  127. storage = other.storage;
  128. start = other.start;
  129. return *this;
  130. }
  131. };
  132. //This type is the first part of the structure controlling a short string
  133. //The "data" member stores
  134. struct short_header
  135. {
  136. unsigned char is_short : 1;
  137. unsigned char length : (CHAR_BIT - 1);
  138. };
  139. //This type has the same alignment and size as long_t but it's POD
  140. //so, unlike long_t, it can be placed in a union
  141. typedef typename container_detail::aligned_storage
  142. <sizeof(long_t), container_detail::alignment_of<long_t>::value>::type long_raw_t;
  143. protected:
  144. static const size_type MinInternalBufferChars = 8;
  145. static const size_type AlignmentOfValueType =
  146. alignment_of<value_type>::value;
  147. static const size_type ShortDataOffset = ((sizeof(short_header)-1)/AlignmentOfValueType+1)*AlignmentOfValueType;
  148. static const size_type ZeroCostInternalBufferChars =
  149. (sizeof(long_t) - ShortDataOffset)/sizeof(value_type);
  150. static const size_type UnalignedFinalInternalBufferChars =
  151. (ZeroCostInternalBufferChars > MinInternalBufferChars) ?
  152. ZeroCostInternalBufferChars : MinInternalBufferChars;
  153. struct short_t
  154. {
  155. short_header h;
  156. value_type data[UnalignedFinalInternalBufferChars];
  157. };
  158. union repr_t
  159. {
  160. long_raw_t r;
  161. short_t s;
  162. const short_t &short_repr() const
  163. { return s; }
  164. const long_t &long_repr() const
  165. { return *static_cast<const long_t*>(static_cast<const void*>(&r)); }
  166. short_t &short_repr()
  167. { return s; }
  168. long_t &long_repr()
  169. { return *static_cast<long_t*>(static_cast<void*>(&r)); }
  170. };
  171. struct members_holder
  172. : public Allocator
  173. {
  174. members_holder()
  175. : Allocator()
  176. {}
  177. template<class AllocatorConvertible>
  178. explicit members_holder(BOOST_FWD_REF(AllocatorConvertible) a)
  179. : Allocator(boost::forward<AllocatorConvertible>(a))
  180. {}
  181. repr_t m_repr;
  182. } members_;
  183. const Allocator &alloc() const
  184. { return members_; }
  185. Allocator &alloc()
  186. { return members_; }
  187. static const size_type InternalBufferChars = (sizeof(repr_t) - ShortDataOffset)/sizeof(value_type);
  188. private:
  189. static const size_type MinAllocation = InternalBufferChars*2;
  190. protected:
  191. bool is_short() const
  192. { return static_cast<bool>(this->members_.m_repr.s.h.is_short != 0); }
  193. void is_short(bool yes)
  194. {
  195. const bool was_short = this->is_short();
  196. if(yes && !was_short){
  197. allocator_traits_type::destroy
  198. ( this->alloc()
  199. , static_cast<long_t*>(static_cast<void*>(&this->members_.m_repr.r))
  200. );
  201. this->members_.m_repr.s.h.is_short = true;
  202. }
  203. else if(!yes && was_short){
  204. allocator_traits_type::construct
  205. ( this->alloc()
  206. , static_cast<long_t*>(static_cast<void*>(&this->members_.m_repr.r))
  207. );
  208. this->members_.m_repr.s.h.is_short = false;
  209. }
  210. }
  211. private:
  212. void init()
  213. {
  214. this->members_.m_repr.s.h.is_short = 1;
  215. this->members_.m_repr.s.h.length = 0;
  216. }
  217. protected:
  218. typedef container_detail::integral_constant<unsigned,
  219. boost::container::container_detail::version<Allocator>::value> alloc_version;
  220. pointer allocation_command(allocation_type command,
  221. size_type limit_size,
  222. size_type &prefer_in_recvd_out_size,
  223. pointer &reuse)
  224. {
  225. if(this->is_short() && (command & (expand_fwd | expand_bwd)) ){
  226. reuse = 0;
  227. command &= ~(expand_fwd | expand_bwd);
  228. }
  229. return container_detail::allocator_version_traits<Allocator>::allocation_command
  230. (this->alloc(), command, limit_size, prefer_in_recvd_out_size, reuse);
  231. }
  232. size_type next_capacity(size_type additional_objects) const
  233. {
  234. return next_capacity_calculator
  235. <size_type, NextCapacityDouble /*NextCapacity60Percent*/>::
  236. get( allocator_traits_type::max_size(this->alloc())
  237. , this->priv_storage(), additional_objects );
  238. }
  239. void deallocate(pointer p, size_type n)
  240. {
  241. if (p && (n > InternalBufferChars))
  242. this->alloc().deallocate(p, n);
  243. }
  244. void construct(pointer p, const value_type &value = value_type())
  245. {
  246. allocator_traits_type::construct
  247. ( this->alloc()
  248. , container_detail::to_raw_pointer(p)
  249. , value
  250. );
  251. }
  252. void destroy(pointer p, size_type n)
  253. {
  254. value_type *raw_p = container_detail::to_raw_pointer(p);
  255. for(; n--; ++raw_p){
  256. allocator_traits_type::destroy( this->alloc(), raw_p);
  257. }
  258. }
  259. void destroy(pointer p)
  260. {
  261. allocator_traits_type::destroy
  262. ( this->alloc()
  263. , container_detail::to_raw_pointer(p)
  264. );
  265. }
  266. void allocate_initial_block(size_type n)
  267. {
  268. if (n <= this->max_size()) {
  269. if(n > InternalBufferChars){
  270. size_type new_cap = this->next_capacity(n);
  271. pointer reuse = 0;
  272. pointer p = this->allocation_command(allocate_new, n, new_cap, reuse);
  273. this->is_short(false);
  274. this->priv_long_addr(p);
  275. this->priv_long_size(0);
  276. this->priv_storage(new_cap);
  277. }
  278. }
  279. else{
  280. throw_length_error("basic_string::allocate_initial_block max_size() exceeded");
  281. }
  282. }
  283. void deallocate_block()
  284. { this->deallocate(this->priv_addr(), this->priv_storage()); }
  285. size_type max_size() const
  286. { return allocator_traits_type::max_size(this->alloc()) - 1; }
  287. protected:
  288. size_type priv_capacity() const
  289. { return this->priv_storage() - 1; }
  290. pointer priv_short_addr() const
  291. { return pointer_traits::pointer_to(const_cast<value_type&>(this->members_.m_repr.short_repr().data[0])); }
  292. pointer priv_long_addr() const
  293. { return this->members_.m_repr.long_repr().start; }
  294. pointer priv_addr() const
  295. {
  296. return this->is_short()
  297. ? priv_short_addr()
  298. : priv_long_addr()
  299. ;
  300. }
  301. pointer priv_end_addr() const
  302. {
  303. return this->is_short()
  304. ? this->priv_short_addr() + this->priv_short_size()
  305. : this->priv_long_addr() + this->priv_long_size()
  306. ;
  307. }
  308. void priv_long_addr(pointer addr)
  309. { this->members_.m_repr.long_repr().start = addr; }
  310. size_type priv_storage() const
  311. { return this->is_short() ? priv_short_storage() : priv_long_storage(); }
  312. size_type priv_short_storage() const
  313. { return InternalBufferChars; }
  314. size_type priv_long_storage() const
  315. { return this->members_.m_repr.long_repr().storage; }
  316. void priv_storage(size_type storage)
  317. {
  318. if(!this->is_short())
  319. this->priv_long_storage(storage);
  320. }
  321. void priv_long_storage(size_type storage)
  322. {
  323. this->members_.m_repr.long_repr().storage = storage;
  324. }
  325. size_type priv_size() const
  326. { return this->is_short() ? this->priv_short_size() : this->priv_long_size(); }
  327. size_type priv_short_size() const
  328. { return this->members_.m_repr.short_repr().h.length; }
  329. size_type priv_long_size() const
  330. { return this->members_.m_repr.long_repr().length; }
  331. void priv_size(size_type sz)
  332. {
  333. if(this->is_short())
  334. this->priv_short_size(sz);
  335. else
  336. this->priv_long_size(sz);
  337. }
  338. void priv_short_size(size_type sz)
  339. {
  340. this->members_.m_repr.s.h.length = (unsigned char)sz;
  341. }
  342. void priv_long_size(size_type sz)
  343. {
  344. this->members_.m_repr.long_repr().length = sz;
  345. }
  346. void swap_data(basic_string_base& other)
  347. {
  348. if(this->is_short()){
  349. if(other.is_short()){
  350. repr_t tmp(this->members_.m_repr);
  351. this->members_.m_repr = other.members_.m_repr;
  352. other.members_.m_repr = tmp;
  353. }
  354. else{
  355. short_t short_backup(this->members_.m_repr.short_repr());
  356. this->members_.m_repr.short_repr().~short_t();
  357. ::new(&this->members_.m_repr.long_repr()) long_t(other.members_.m_repr.long_repr());
  358. other.members_.m_repr.long_repr().~long_t();
  359. ::new(&other.members_.m_repr.short_repr()) short_t(short_backup);
  360. }
  361. }
  362. else{
  363. if(other.is_short()){
  364. short_t short_backup(other.members_.m_repr.short_repr());
  365. other.members_.m_repr.short_repr().~short_t();
  366. ::new(&other.members_.m_repr.long_repr()) long_t(this->members_.m_repr.long_repr());
  367. this->members_.m_repr.long_repr().~long_t();
  368. ::new(&this->members_.m_repr.short_repr()) short_t(short_backup);
  369. }
  370. else{
  371. boost::adl_move_swap(this->members_.m_repr.long_repr(), other.members_.m_repr.long_repr());
  372. }
  373. }
  374. }
  375. };
  376. } //namespace container_detail {
  377. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  378. //! The basic_string class represents a Sequence of characters. It contains all the
  379. //! usual operations of a Sequence, and, additionally, it contains standard string
  380. //! operations such as search and concatenation.
  381. //!
  382. //! The basic_string class is parameterized by character type, and by that type's
  383. //! Character Traits.
  384. //!
  385. //! This class has performance characteristics very much like vector<>, meaning,
  386. //! for example, that it does not perform reference-count or copy-on-write, and that
  387. //! concatenation of two strings is an O(N) operation.
  388. //!
  389. //! Some of basic_string's member functions use an unusual method of specifying positions
  390. //! and ranges. In addition to the conventional method using iterators, many of
  391. //! basic_string's member functions use a single value pos of type size_type to represent a
  392. //! position (in which case the position is begin() + pos, and many of basic_string's
  393. //! member functions use two values, pos and n, to represent a range. In that case pos is
  394. //! the beginning of the range and n is its size. That is, the range is
  395. //! [begin() + pos, begin() + pos + n).
  396. //!
  397. //! Note that the C++ standard does not specify the complexity of basic_string operations.
  398. //! In this implementation, basic_string has performance characteristics very similar to
  399. //! those of vector: access to a single character is O(1), while copy and concatenation
  400. //! are O(N).
  401. //!
  402. //! In this implementation, begin(),
  403. //! end(), rbegin(), rend(), operator[], c_str(), and data() do not invalidate iterators.
  404. //! In this implementation, iterators are only invalidated by member functions that
  405. //! explicitly change the string's contents.
  406. //!
  407. //! \tparam CharT The type of character it contains.
  408. //! \tparam Traits The Character Traits type, which encapsulates basic character operations
  409. //! \tparam Allocator The allocator, used for internal memory management.
  410. #ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  411. template <class CharT, class Traits = std::char_traits<CharT>, class Allocator = new_allocator<CharT> >
  412. #else
  413. template <class CharT, class Traits, class Allocator>
  414. #endif
  415. class basic_string
  416. : private container_detail::basic_string_base<Allocator>
  417. {
  418. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  419. private:
  420. typedef allocator_traits<Allocator> allocator_traits_type;
  421. BOOST_COPYABLE_AND_MOVABLE(basic_string)
  422. typedef container_detail::basic_string_base<Allocator> base_t;
  423. static const typename base_t::size_type InternalBufferChars = base_t::InternalBufferChars;
  424. protected:
  425. // Allocator helper class to use a char_traits as a function object.
  426. template <class Tr>
  427. struct Eq_traits
  428. {
  429. //Compatibility with std::binary_function
  430. typedef typename Tr::char_type first_argument_type;
  431. typedef typename Tr::char_type second_argument_type;
  432. typedef bool result_type;
  433. bool operator()(const first_argument_type& x, const second_argument_type& y) const
  434. { return Tr::eq(x, y); }
  435. };
  436. template <class Tr>
  437. struct Not_within_traits
  438. {
  439. typedef typename Tr::char_type argument_type;
  440. typedef bool result_type;
  441. typedef const typename Tr::char_type* Pointer;
  442. const Pointer m_first;
  443. const Pointer m_last;
  444. Not_within_traits(Pointer f, Pointer l)
  445. : m_first(f), m_last(l) {}
  446. bool operator()(const typename Tr::char_type& x) const
  447. {
  448. return std::find_if(m_first, m_last,
  449. std::bind1st(Eq_traits<Tr>(), x)) == m_last;
  450. }
  451. };
  452. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  453. public:
  454. //////////////////////////////////////////////
  455. //
  456. // types
  457. //
  458. //////////////////////////////////////////////
  459. typedef Traits traits_type;
  460. typedef CharT value_type;
  461. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  462. typedef typename ::boost::container::allocator_traits<Allocator>::const_pointer const_pointer;
  463. typedef typename ::boost::container::allocator_traits<Allocator>::reference reference;
  464. typedef typename ::boost::container::allocator_traits<Allocator>::const_reference const_reference;
  465. typedef typename ::boost::container::allocator_traits<Allocator>::size_type size_type;
  466. typedef typename ::boost::container::allocator_traits<Allocator>::difference_type difference_type;
  467. typedef Allocator allocator_type;
  468. typedef BOOST_CONTAINER_IMPDEF(allocator_type) stored_allocator_type;
  469. typedef BOOST_CONTAINER_IMPDEF(pointer) iterator;
  470. typedef BOOST_CONTAINER_IMPDEF(const_pointer) const_iterator;
  471. typedef BOOST_CONTAINER_IMPDEF(boost::container::reverse_iterator<iterator>) reverse_iterator;
  472. typedef BOOST_CONTAINER_IMPDEF(boost::container::reverse_iterator<const_iterator>) const_reverse_iterator;
  473. static const size_type npos = size_type(-1);
  474. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  475. private:
  476. typedef constant_iterator<CharT, difference_type> cvalue_iterator;
  477. typedef typename base_t::alloc_version alloc_version;
  478. typedef ::boost::intrusive::pointer_traits<pointer> pointer_traits;
  479. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  480. public: // Constructor, destructor, assignment.
  481. //////////////////////////////////////////////
  482. //
  483. // construct/copy/destroy
  484. //
  485. //////////////////////////////////////////////
  486. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  487. struct reserve_t {};
  488. basic_string(reserve_t, size_type n,
  489. const allocator_type& a = allocator_type())
  490. //Select allocator as in copy constructor as reserve_t-based constructors
  491. //are two step copies optimized for capacity
  492. : base_t( allocator_traits_type::select_on_container_copy_construction(a)
  493. , n + 1)
  494. { this->priv_terminate_string(); }
  495. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  496. //! <b>Effects</b>: Default constructs a basic_string.
  497. //!
  498. //! <b>Throws</b>: If allocator_type's default constructor throws.
  499. basic_string()
  500. : base_t()
  501. { this->priv_terminate_string(); }
  502. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter.
  503. //!
  504. //! <b>Throws</b>: Nothing
  505. explicit basic_string(const allocator_type& a) BOOST_NOEXCEPT_OR_NOTHROW
  506. : base_t(a)
  507. { this->priv_terminate_string(); }
  508. //! <b>Effects</b>: Copy constructs a basic_string.
  509. //!
  510. //! <b>Postcondition</b>: x == *this.
  511. //!
  512. //! <b>Throws</b>: If allocator_type's default constructor or allocation throws.
  513. basic_string(const basic_string& s)
  514. : base_t(allocator_traits_type::select_on_container_copy_construction(s.alloc()))
  515. {
  516. this->priv_terminate_string();
  517. this->assign(s.begin(), s.end());
  518. }
  519. //! <b>Effects</b>: Move constructor. Moves s's resources to *this.
  520. //!
  521. //! <b>Throws</b>: Nothing.
  522. //!
  523. //! <b>Complexity</b>: Constant.
  524. basic_string(BOOST_RV_REF(basic_string) s) BOOST_NOEXCEPT_OR_NOTHROW
  525. : base_t(boost::move(s.alloc()))
  526. {
  527. if(s.alloc() == this->alloc()){
  528. this->swap_data(s);
  529. }
  530. else{
  531. this->assign(s.begin(), s.end());
  532. }
  533. }
  534. //! <b>Effects</b>: Copy constructs a basic_string using the specified allocator.
  535. //!
  536. //! <b>Postcondition</b>: x == *this.
  537. //!
  538. //! <b>Throws</b>: If allocation throws.
  539. basic_string(const basic_string& s, const allocator_type &a)
  540. : base_t(a)
  541. {
  542. this->priv_terminate_string();
  543. this->assign(s.begin(), s.end());
  544. }
  545. //! <b>Effects</b>: Move constructor using the specified allocator.
  546. //! Moves s's resources to *this.
  547. //!
  548. //! <b>Throws</b>: If allocation throws.
  549. //!
  550. //! <b>Complexity</b>: Constant if a == s.get_allocator(), linear otherwise.
  551. basic_string(BOOST_RV_REF(basic_string) s, const allocator_type &a)
  552. : base_t(a)
  553. {
  554. this->priv_terminate_string();
  555. if(a == this->alloc()){
  556. this->swap_data(s);
  557. }
  558. else{
  559. this->assign(s.begin(), s.end());
  560. }
  561. }
  562. //! <b>Effects</b>: Constructs a basic_string with a default-constructed allocator,
  563. //! and is initialized by a specific number of characters of the s string.
  564. basic_string(const basic_string& s, size_type pos, size_type n = npos)
  565. : base_t()
  566. {
  567. this->priv_terminate_string();
  568. if (pos > s.size())
  569. throw_out_of_range("basic_string::basic_string out of range position");
  570. else
  571. this->assign
  572. (s.begin() + pos, s.begin() + pos + container_detail::min_value(n, s.size() - pos));
  573. }
  574. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  575. //! and is initialized by a specific number of characters of the s string.
  576. basic_string(const basic_string& s, size_type pos, size_type n, const allocator_type& a)
  577. : base_t(a)
  578. {
  579. this->priv_terminate_string();
  580. if (pos > s.size())
  581. throw_out_of_range("basic_string::basic_string out of range position");
  582. else
  583. this->assign
  584. (s.begin() + pos, s.begin() + pos + container_detail::min_value(n, s.size() - pos));
  585. }
  586. //! <b>Effects</b>: Constructs a basic_string taking a default-constructed allocator,
  587. //! and is initialized by a specific number of characters of the s c-string.
  588. basic_string(const CharT* s, size_type n)
  589. : base_t()
  590. {
  591. this->priv_terminate_string();
  592. this->assign(s, s + n);
  593. }
  594. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  595. //! and is initialized by a specific number of characters of the s c-string.
  596. basic_string(const CharT* s, size_type n, const allocator_type& a)
  597. : base_t(a)
  598. {
  599. this->priv_terminate_string();
  600. this->assign(s, s + n);
  601. }
  602. //! <b>Effects</b>: Constructs a basic_string with a default-constructed allocator,
  603. //! and is initialized by the null-terminated s c-string.
  604. basic_string(const CharT* s)
  605. : base_t()
  606. {
  607. this->priv_terminate_string();
  608. this->assign(s, s + Traits::length(s));
  609. }
  610. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  611. //! and is initialized by the null-terminated s c-string.
  612. basic_string(const CharT* s, const allocator_type& a)
  613. : base_t(a)
  614. {
  615. this->priv_terminate_string();
  616. this->assign(s, s + Traits::length(s));
  617. }
  618. //! <b>Effects</b>: Constructs a basic_string with a default-constructed allocator,
  619. //! and is initialized by n copies of c.
  620. basic_string(size_type n, CharT c)
  621. : base_t()
  622. {
  623. this->priv_terminate_string();
  624. this->assign(n, c);
  625. }
  626. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  627. //! and is initialized by n copies of c.
  628. basic_string(size_type n, CharT c, const allocator_type& a)
  629. : base_t(a)
  630. {
  631. this->priv_terminate_string();
  632. this->assign(n, c);
  633. }
  634. //! <b>Effects</b>: Constructs a basic_string with a default-constructed allocator,
  635. //! and is initialized by n default-initialized characters.
  636. basic_string(size_type n, default_init_t)
  637. : base_t(n + 1)
  638. {
  639. this->priv_size(n);
  640. this->priv_terminate_string();
  641. }
  642. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  643. //! and is initialized by n default-initialized characters.
  644. basic_string(size_type n, default_init_t, const allocator_type& a)
  645. : base_t(a, n + 1)
  646. {
  647. this->priv_size(n);
  648. this->priv_terminate_string();
  649. }
  650. //! <b>Effects</b>: Constructs a basic_string with a default-constructed allocator,
  651. //! and a range of iterators.
  652. template <class InputIterator>
  653. basic_string(InputIterator f, InputIterator l)
  654. : base_t()
  655. {
  656. this->priv_terminate_string();
  657. this->assign(f, l);
  658. }
  659. //! <b>Effects</b>: Constructs a basic_string taking the allocator as parameter,
  660. //! and a range of iterators.
  661. template <class InputIterator>
  662. basic_string(InputIterator f, InputIterator l, const allocator_type& a)
  663. : base_t(a)
  664. {
  665. this->priv_terminate_string();
  666. this->assign(f, l);
  667. }
  668. //! <b>Effects</b>: Destroys the basic_string. All used memory is deallocated.
  669. //!
  670. //! <b>Throws</b>: Nothing.
  671. //!
  672. //! <b>Complexity</b>: Constant.
  673. ~basic_string() BOOST_NOEXCEPT_OR_NOTHROW
  674. {}
  675. //! <b>Effects</b>: Copy constructs a string.
  676. //!
  677. //! <b>Postcondition</b>: x == *this.
  678. //!
  679. //! <b>Complexity</b>: Linear to the elements x contains.
  680. basic_string& operator=(BOOST_COPY_ASSIGN_REF(basic_string) x)
  681. {
  682. if (&x != this){
  683. allocator_type &this_alloc = this->alloc();
  684. const allocator_type &x_alloc = x.alloc();
  685. container_detail::bool_<allocator_traits_type::
  686. propagate_on_container_copy_assignment::value> flag;
  687. if(flag && this_alloc != x_alloc){
  688. if(!this->is_short()){
  689. this->deallocate_block();
  690. this->is_short(true);
  691. Traits::assign(*this->priv_addr(), CharT(0));
  692. this->priv_short_size(0);
  693. }
  694. }
  695. container_detail::assign_alloc(this->alloc(), x.alloc(), flag);
  696. this->assign(x.begin(), x.end());
  697. }
  698. return *this;
  699. }
  700. //! <b>Effects</b>: Move constructor. Moves x's resources to *this.
  701. //!
  702. //! <b>Throws</b>: If allocator_traits_type::propagate_on_container_move_assignment
  703. //! is false and allocation throws
  704. //!
  705. //! <b>Complexity</b>: Constant if allocator_traits_type::
  706. //! propagate_on_container_move_assignment is true or
  707. //! this->get>allocator() == x.get_allocator(). Linear otherwise.
  708. basic_string& operator=(BOOST_RV_REF(basic_string) x)
  709. BOOST_NOEXCEPT_IF(allocator_traits_type::propagate_on_container_move_assignment::value
  710. || allocator_traits_type::is_always_equal::value)
  711. {
  712. //for move constructor, no aliasing (&x != this) is assummed.
  713. BOOST_ASSERT(this != &x);
  714. allocator_type &this_alloc = this->alloc();
  715. allocator_type &x_alloc = x.alloc();
  716. const bool propagate_alloc = allocator_traits_type::
  717. propagate_on_container_move_assignment::value;
  718. container_detail::bool_<propagate_alloc> flag;
  719. const bool allocators_equal = this_alloc == x_alloc; (void)allocators_equal;
  720. //Resources can be transferred if both allocators are
  721. //going to be equal after this function (either propagated or already equal)
  722. if(propagate_alloc || allocators_equal){
  723. //Destroy objects but retain memory in case x reuses it in the future
  724. this->clear();
  725. //Move allocator if needed
  726. container_detail::move_alloc(this_alloc, x_alloc, flag);
  727. //Nothrow swap
  728. this->swap_data(x);
  729. }
  730. //Else do a one by one move
  731. else{
  732. this->assign( x.begin(), x.end());
  733. }
  734. return *this;
  735. }
  736. //! <b>Effects</b>: Assignment from a null-terminated c-string.
  737. basic_string& operator=(const CharT* s)
  738. { return this->assign(s, s + Traits::length(s)); }
  739. //! <b>Effects</b>: Assignment from character.
  740. basic_string& operator=(CharT c)
  741. { return this->assign(static_cast<size_type>(1), c); }
  742. //! <b>Effects</b>: Returns a copy of the internal allocator.
  743. //!
  744. //! <b>Throws</b>: If allocator's copy constructor throws.
  745. //!
  746. //! <b>Complexity</b>: Constant.
  747. allocator_type get_allocator() const BOOST_NOEXCEPT_OR_NOTHROW
  748. { return this->alloc(); }
  749. //! <b>Effects</b>: Returns a reference to the internal allocator.
  750. //!
  751. //! <b>Throws</b>: Nothing
  752. //!
  753. //! <b>Complexity</b>: Constant.
  754. //!
  755. //! <b>Note</b>: Non-standard extension.
  756. stored_allocator_type &get_stored_allocator() BOOST_NOEXCEPT_OR_NOTHROW
  757. { return this->alloc(); }
  758. //! <b>Effects</b>: Returns a reference to the internal allocator.
  759. //!
  760. //! <b>Throws</b>: Nothing
  761. //!
  762. //! <b>Complexity</b>: Constant.
  763. //!
  764. //! <b>Note</b>: Non-standard extension.
  765. const stored_allocator_type &get_stored_allocator() const BOOST_NOEXCEPT_OR_NOTHROW
  766. { return this->alloc(); }
  767. //////////////////////////////////////////////
  768. //
  769. // iterators
  770. //
  771. //////////////////////////////////////////////
  772. //! <b>Effects</b>: Returns an iterator to the first element contained in the vector.
  773. //!
  774. //! <b>Throws</b>: Nothing.
  775. //!
  776. //! <b>Complexity</b>: Constant.
  777. iterator begin() BOOST_NOEXCEPT_OR_NOTHROW
  778. { return this->priv_addr(); }
  779. //! <b>Effects</b>: Returns a const_iterator to the first element contained in the vector.
  780. //!
  781. //! <b>Throws</b>: Nothing.
  782. //!
  783. //! <b>Complexity</b>: Constant.
  784. const_iterator begin() const BOOST_NOEXCEPT_OR_NOTHROW
  785. { return this->priv_addr(); }
  786. //! <b>Effects</b>: Returns an iterator to the end of the vector.
  787. //!
  788. //! <b>Throws</b>: Nothing.
  789. //!
  790. //! <b>Complexity</b>: Constant.
  791. iterator end() BOOST_NOEXCEPT_OR_NOTHROW
  792. { return this->priv_end_addr(); }
  793. //! <b>Effects</b>: Returns a const_iterator to the end of the vector.
  794. //!
  795. //! <b>Throws</b>: Nothing.
  796. //!
  797. //! <b>Complexity</b>: Constant.
  798. const_iterator end() const BOOST_NOEXCEPT_OR_NOTHROW
  799. { return this->priv_end_addr(); }
  800. //! <b>Effects</b>: Returns a reverse_iterator pointing to the beginning
  801. //! of the reversed vector.
  802. //!
  803. //! <b>Throws</b>: Nothing.
  804. //!
  805. //! <b>Complexity</b>: Constant.
  806. reverse_iterator rbegin() BOOST_NOEXCEPT_OR_NOTHROW
  807. { return reverse_iterator(this->priv_end_addr()); }
  808. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the beginning
  809. //! of the reversed vector.
  810. //!
  811. //! <b>Throws</b>: Nothing.
  812. //!
  813. //! <b>Complexity</b>: Constant.
  814. const_reverse_iterator rbegin() const BOOST_NOEXCEPT_OR_NOTHROW
  815. { return this->crbegin(); }
  816. //! <b>Effects</b>: Returns a reverse_iterator pointing to the end
  817. //! of the reversed vector.
  818. //!
  819. //! <b>Throws</b>: Nothing.
  820. //!
  821. //! <b>Complexity</b>: Constant.
  822. reverse_iterator rend() BOOST_NOEXCEPT_OR_NOTHROW
  823. { return reverse_iterator(this->priv_addr()); }
  824. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the end
  825. //! of the reversed vector.
  826. //!
  827. //! <b>Throws</b>: Nothing.
  828. //!
  829. //! <b>Complexity</b>: Constant.
  830. const_reverse_iterator rend() const BOOST_NOEXCEPT_OR_NOTHROW
  831. { return this->crend(); }
  832. //! <b>Effects</b>: Returns a const_iterator to the first element contained in the vector.
  833. //!
  834. //! <b>Throws</b>: Nothing.
  835. //!
  836. //! <b>Complexity</b>: Constant.
  837. const_iterator cbegin() const BOOST_NOEXCEPT_OR_NOTHROW
  838. { return this->priv_addr(); }
  839. //! <b>Effects</b>: Returns a const_iterator to the end of the vector.
  840. //!
  841. //! <b>Throws</b>: Nothing.
  842. //!
  843. //! <b>Complexity</b>: Constant.
  844. const_iterator cend() const BOOST_NOEXCEPT_OR_NOTHROW
  845. { return this->priv_end_addr(); }
  846. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the beginning
  847. //! of the reversed vector.
  848. //!
  849. //! <b>Throws</b>: Nothing.
  850. //!
  851. //! <b>Complexity</b>: Constant.
  852. const_reverse_iterator crbegin() const BOOST_NOEXCEPT_OR_NOTHROW
  853. { return const_reverse_iterator(this->priv_end_addr()); }
  854. //! <b>Effects</b>: Returns a const_reverse_iterator pointing to the end
  855. //! of the reversed vector.
  856. //!
  857. //! <b>Throws</b>: Nothing.
  858. //!
  859. //! <b>Complexity</b>: Constant.
  860. const_reverse_iterator crend() const BOOST_NOEXCEPT_OR_NOTHROW
  861. { return const_reverse_iterator(this->priv_addr()); }
  862. //////////////////////////////////////////////
  863. //
  864. // capacity
  865. //
  866. //////////////////////////////////////////////
  867. //! <b>Effects</b>: Returns true if the vector contains no elements.
  868. //!
  869. //! <b>Throws</b>: Nothing.
  870. //!
  871. //! <b>Complexity</b>: Constant.
  872. bool empty() const BOOST_NOEXCEPT_OR_NOTHROW
  873. { return !this->priv_size(); }
  874. //! <b>Effects</b>: Returns the number of the elements contained in the vector.
  875. //!
  876. //! <b>Throws</b>: Nothing.
  877. //!
  878. //! <b>Complexity</b>: Constant.
  879. size_type size() const BOOST_NOEXCEPT_OR_NOTHROW
  880. { return this->priv_size(); }
  881. //! <b>Effects</b>: Returns the number of the elements contained in the vector.
  882. //!
  883. //! <b>Throws</b>: Nothing.
  884. //!
  885. //! <b>Complexity</b>: Constant.
  886. size_type length() const BOOST_NOEXCEPT_OR_NOTHROW
  887. { return this->size(); }
  888. //! <b>Effects</b>: Returns the largest possible size of the vector.
  889. //!
  890. //! <b>Throws</b>: Nothing.
  891. //!
  892. //! <b>Complexity</b>: Constant.
  893. size_type max_size() const BOOST_NOEXCEPT_OR_NOTHROW
  894. { return base_t::max_size(); }
  895. //! <b>Effects</b>: Inserts or erases elements at the end such that
  896. //! the size becomes n. New elements are copy constructed from x.
  897. //!
  898. //! <b>Throws</b>: If memory allocation throws
  899. //!
  900. //! <b>Complexity</b>: Linear to the difference between size() and new_size.
  901. void resize(size_type n, CharT c)
  902. {
  903. if (n <= this->size())
  904. this->erase(this->begin() + n, this->end());
  905. else
  906. this->append(n - this->size(), c);
  907. }
  908. //! <b>Effects</b>: Inserts or erases elements at the end such that
  909. //! the size becomes n. New elements are value initialized.
  910. //!
  911. //! <b>Throws</b>: If memory allocation throws
  912. //!
  913. //! <b>Complexity</b>: Linear to the difference between size() and new_size.
  914. void resize(size_type n)
  915. { resize(n, CharT()); }
  916. //! <b>Effects</b>: Inserts or erases elements at the end such that
  917. //! the size becomes n. New elements are uninitialized.
  918. //!
  919. //! <b>Throws</b>: If memory allocation throws
  920. //!
  921. //! <b>Complexity</b>: Linear to the difference between size() and new_size.
  922. //!
  923. //! <b>Note</b>: Non-standard extension
  924. void resize(size_type n, default_init_t)
  925. {
  926. if (n <= this->size())
  927. this->erase(this->begin() + n, this->end());
  928. else{
  929. this->priv_reserve(n, false);
  930. this->priv_size(n);
  931. this->priv_terminate_string();
  932. }
  933. }
  934. //! <b>Effects</b>: Number of elements for which memory has been allocated.
  935. //! capacity() is always greater than or equal to size().
  936. //!
  937. //! <b>Throws</b>: Nothing.
  938. //!
  939. //! <b>Complexity</b>: Constant.
  940. size_type capacity() const BOOST_NOEXCEPT_OR_NOTHROW
  941. { return this->priv_capacity(); }
  942. //! <b>Effects</b>: If n is less than or equal to capacity(), this call has no
  943. //! effect. Otherwise, it is a request for allocation of additional memory.
  944. //! If the request is successful, then capacity() is greater than or equal to
  945. //! n; otherwise, capacity() is unchanged. In either case, size() is unchanged.
  946. //!
  947. //! <b>Throws</b>: If memory allocation allocation throws
  948. void reserve(size_type res_arg)
  949. { this->priv_reserve(res_arg); }
  950. //! <b>Effects</b>: Tries to deallocate the excess of memory created
  951. //! with previous allocations. The size of the string is unchanged
  952. //!
  953. //! <b>Throws</b>: Nothing
  954. //!
  955. //! <b>Complexity</b>: Linear to size().
  956. void shrink_to_fit()
  957. {
  958. //Check if shrinking is possible
  959. if(this->priv_storage() > InternalBufferChars){
  960. //Check if we should pass from dynamically allocated buffer
  961. //to the internal storage
  962. if(this->priv_size() < InternalBufferChars){
  963. //Dynamically allocated buffer attributes
  964. pointer long_addr = this->priv_long_addr();
  965. size_type long_storage = this->priv_long_storage();
  966. size_type long_size = this->priv_long_size();
  967. //Shrink from allocated buffer to the internal one, including trailing null
  968. Traits::copy( container_detail::to_raw_pointer(this->priv_short_addr())
  969. , container_detail::to_raw_pointer(long_addr)
  970. , long_size+1);
  971. this->is_short(true);
  972. this->alloc().deallocate(long_addr, long_storage);
  973. }
  974. else{
  975. //Shrinking in dynamic buffer
  976. this->priv_shrink_to_fit_dynamic_buffer(alloc_version());
  977. }
  978. }
  979. }
  980. //////////////////////////////////////////////
  981. //
  982. // element access
  983. //
  984. //////////////////////////////////////////////
  985. //! <b>Requires</b>: !empty()
  986. //!
  987. //! <b>Effects</b>: Returns a reference to the first
  988. //! element of the container.
  989. //!
  990. //! <b>Throws</b>: Nothing.
  991. //!
  992. //! <b>Complexity</b>: Constant.
  993. reference front() BOOST_NOEXCEPT_OR_NOTHROW
  994. {
  995. BOOST_ASSERT(!this->empty());
  996. return *this->priv_addr();
  997. }
  998. //! <b>Requires</b>: !empty()
  999. //!
  1000. //! <b>Effects</b>: Returns a const reference to the first
  1001. //! element of the container.
  1002. //!
  1003. //! <b>Throws</b>: Nothing.
  1004. //!
  1005. //! <b>Complexity</b>: Constant.
  1006. const_reference front() const BOOST_NOEXCEPT_OR_NOTHROW
  1007. {
  1008. BOOST_ASSERT(!this->empty());
  1009. return *this->priv_addr();
  1010. }
  1011. //! <b>Requires</b>: !empty()
  1012. //!
  1013. //! <b>Effects</b>: Returns a reference to the last
  1014. //! element of the container.
  1015. //!
  1016. //! <b>Throws</b>: Nothing.
  1017. //!
  1018. //! <b>Complexity</b>: Constant.
  1019. reference back() BOOST_NOEXCEPT_OR_NOTHROW
  1020. {
  1021. BOOST_ASSERT(!this->empty());
  1022. return *(this->priv_addr() + (this->size() - 1u) );
  1023. }
  1024. //! <b>Requires</b>: !empty()
  1025. //!
  1026. //! <b>Effects</b>: Returns a const reference to the last
  1027. //! element of the container.
  1028. //!
  1029. //! <b>Throws</b>: Nothing.
  1030. //!
  1031. //! <b>Complexity</b>: Constant.
  1032. const_reference back() const BOOST_NOEXCEPT_OR_NOTHROW
  1033. {
  1034. BOOST_ASSERT(!this->empty());
  1035. return *(this->priv_addr() + (this->size() - 1u) );
  1036. }
  1037. //! <b>Requires</b>: size() > n.
  1038. //!
  1039. //! <b>Effects</b>: Returns a reference to the nth element
  1040. //! from the beginning of the container.
  1041. //!
  1042. //! <b>Throws</b>: Nothing.
  1043. //!
  1044. //! <b>Complexity</b>: Constant.
  1045. reference operator[](size_type n) BOOST_NOEXCEPT_OR_NOTHROW
  1046. {
  1047. BOOST_ASSERT(this->size() > n);
  1048. return *(this->priv_addr() + n);
  1049. }
  1050. //! <b>Requires</b>: size() > n.
  1051. //!
  1052. //! <b>Effects</b>: Returns a const reference to the nth element
  1053. //! from the beginning of the container.
  1054. //!
  1055. //! <b>Throws</b>: Nothing.
  1056. //!
  1057. //! <b>Complexity</b>: Constant.
  1058. const_reference operator[](size_type n) const BOOST_NOEXCEPT_OR_NOTHROW
  1059. {
  1060. BOOST_ASSERT(this->size() > n);
  1061. return *(this->priv_addr() + n);
  1062. }
  1063. //! <b>Requires</b>: size() > n.
  1064. //!
  1065. //! <b>Effects</b>: Returns a reference to the nth element
  1066. //! from the beginning of the container.
  1067. //!
  1068. //! <b>Throws</b>: std::range_error if n >= size()
  1069. //!
  1070. //! <b>Complexity</b>: Constant.
  1071. reference at(size_type n)
  1072. {
  1073. if (n >= this->size())
  1074. throw_out_of_range("basic_string::at invalid subscript");
  1075. return *(this->priv_addr() + n);
  1076. }
  1077. //! <b>Requires</b>: size() > n.
  1078. //!
  1079. //! <b>Effects</b>: Returns a const reference to the nth element
  1080. //! from the beginning of the container.
  1081. //!
  1082. //! <b>Throws</b>: std::range_error if n >= size()
  1083. //!
  1084. //! <b>Complexity</b>: Constant.
  1085. const_reference at(size_type n) const {
  1086. if (n >= this->size())
  1087. throw_out_of_range("basic_string::at invalid subscript");
  1088. return *(this->priv_addr() + n);
  1089. }
  1090. //////////////////////////////////////////////
  1091. //
  1092. // modifiers
  1093. //
  1094. //////////////////////////////////////////////
  1095. //! <b>Effects</b>: Calls append(str.data, str.size()).
  1096. //!
  1097. //! <b>Returns</b>: *this
  1098. basic_string& operator+=(const basic_string& s)
  1099. { return this->append(s); }
  1100. //! <b>Effects</b>: Calls append(s).
  1101. //!
  1102. //! <b>Returns</b>: *this
  1103. basic_string& operator+=(const CharT* s)
  1104. { return this->append(s); }
  1105. //! <b>Effects</b>: Calls append(1, c).
  1106. //!
  1107. //! <b>Returns</b>: *this
  1108. basic_string& operator+=(CharT c)
  1109. { this->push_back(c); return *this; }
  1110. //! <b>Effects</b>: Calls append(str.data(), str.size()).
  1111. //!
  1112. //! <b>Returns</b>: *this
  1113. basic_string& append(const basic_string& s)
  1114. { return this->append(s.begin(), s.end()); }
  1115. //! <b>Requires</b>: pos <= str.size()
  1116. //!
  1117. //! <b>Effects</b>: Determines the effective length rlen of the string to append
  1118. //! as the smaller of n and str.size() - pos and calls append(str.data() + pos, rlen).
  1119. //!
  1120. //! <b>Throws</b>: If memory allocation throws and out_of_range if pos > str.size()
  1121. //!
  1122. //! <b>Returns</b>: *this
  1123. basic_string& append(const basic_string& s, size_type pos, size_type n)
  1124. {
  1125. if (pos > s.size())
  1126. throw_out_of_range("basic_string::append out of range position");
  1127. return this->append(s.begin() + pos,
  1128. s.begin() + pos + container_detail::min_value(n, s.size() - pos));
  1129. }
  1130. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  1131. //!
  1132. //! <b>Effects</b>: The function replaces the string controlled by *this with
  1133. //! a string of length size() + n whose irst size() elements are a copy of the
  1134. //! original string controlled by *this and whose remaining
  1135. //! elements are a copy of the initial n elements of s.
  1136. //!
  1137. //! <b>Throws</b>: If memory allocation throws length_error if size() + n > max_size().
  1138. //!
  1139. //! <b>Returns</b>: *this
  1140. basic_string& append(const CharT* s, size_type n)
  1141. { return this->append(s, s + n); }
  1142. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  1143. //!
  1144. //! <b>Effects</b>: Calls append(s, traits::length(s)).
  1145. //!
  1146. //! <b>Returns</b>: *this
  1147. basic_string& append(const CharT* s)
  1148. { return this->append(s, s + Traits::length(s)); }
  1149. //! <b>Effects</b>: Equivalent to append(basic_string(n, c)).
  1150. //!
  1151. //! <b>Returns</b>: *this
  1152. basic_string& append(size_type n, CharT c)
  1153. { return this->append(cvalue_iterator(c, n), cvalue_iterator()); }
  1154. //! <b>Requires</b>: [first,last) is a valid range.
  1155. //!
  1156. //! <b>Effects</b>: Equivalent to append(basic_string(first, last)).
  1157. //!
  1158. //! <b>Returns</b>: *this
  1159. template <class InputIter>
  1160. basic_string& append(InputIter first, InputIter last)
  1161. { this->insert(this->end(), first, last); return *this; }
  1162. //! <b>Effects</b>: Equivalent to append(static_cast<size_type>(1), c).
  1163. void push_back(CharT c)
  1164. {
  1165. const size_type old_size = this->priv_size();
  1166. if (old_size < this->capacity()){
  1167. const pointer addr = this->priv_addr();
  1168. this->priv_construct_null(addr + old_size + 1);
  1169. Traits::assign(addr[old_size], c);
  1170. this->priv_size(old_size+1);
  1171. }
  1172. else{
  1173. //No enough memory, insert a new object at the end
  1174. this->append(size_type(1), c);
  1175. }
  1176. }
  1177. //! <b>Effects</b>: Equivalent to assign(str, 0, npos).
  1178. //!
  1179. //! <b>Returns</b>: *this
  1180. basic_string& assign(const basic_string& s)
  1181. { return this->operator=(s); }
  1182. //! <b>Effects</b>: The function replaces the string controlled by *this
  1183. //! with a string of length str.size() whose elements are a copy of the string
  1184. //! controlled by str. Leaves str in a valid but unspecified state.
  1185. //!
  1186. //! <b>Throws</b>: Nothing
  1187. //!
  1188. //! <b>Returns</b>: *this
  1189. basic_string& assign(BOOST_RV_REF(basic_string) ms) BOOST_NOEXCEPT_OR_NOTHROW
  1190. { return this->swap_data(ms), *this; }
  1191. //! <b>Requires</b>: pos <= str.size()
  1192. //!
  1193. //! <b>Effects</b>: Determines the effective length rlen of the string to assign as
  1194. //! the smaller of n and str.size() - pos and calls assign(str.data() + pos rlen).
  1195. //!
  1196. //! <b>Throws</b>: If memory allocation throws or out_of_range if pos > str.size().
  1197. //!
  1198. //! <b>Returns</b>: *this
  1199. basic_string& assign(const basic_string& s, size_type pos, size_type n)
  1200. {
  1201. if (pos > s.size())
  1202. throw_out_of_range("basic_string::assign out of range position");
  1203. return this->assign(s.begin() + pos,
  1204. s.begin() + pos + container_detail::min_value(n, s.size() - pos));
  1205. }
  1206. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  1207. //!
  1208. //! <b>Effects</b>: Replaces the string controlled by *this with a string of
  1209. //! length n whose elements are a copy of those pointed to by s.
  1210. //!
  1211. //! <b>Throws</b>: If memory allocation throws or length_error if n > max_size().
  1212. //!
  1213. //! <b>Returns</b>: *this
  1214. basic_string& assign(const CharT* s, size_type n)
  1215. { return this->assign(s, s + n); }
  1216. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  1217. //!
  1218. //! <b>Effects</b>: Calls assign(s, traits::length(s)).
  1219. //!
  1220. //! <b>Returns</b>: *this
  1221. basic_string& assign(const CharT* s)
  1222. { return this->assign(s, s + Traits::length(s)); }
  1223. //! <b>Effects</b>: Equivalent to assign(basic_string(n, c)).
  1224. //!
  1225. //! <b>Returns</b>: *this
  1226. basic_string& assign(size_type n, CharT c)
  1227. { return this->assign(cvalue_iterator(c, n), cvalue_iterator()); }
  1228. //! <b>Effects</b>: Equivalent to assign(basic_string(first, last)).
  1229. //!
  1230. //! <b>Returns</b>: *this
  1231. basic_string& assign(const CharT* first, const CharT* last)
  1232. {
  1233. size_type n = static_cast<size_type>(last - first);
  1234. this->reserve(n);
  1235. CharT* ptr = container_detail::to_raw_pointer(this->priv_addr());
  1236. Traits::copy(ptr, first, n);
  1237. this->priv_construct_null(ptr + n);
  1238. this->priv_size(n);
  1239. return *this;
  1240. }
  1241. //! <b>Effects</b>: Equivalent to assign(basic_string(first, last)).
  1242. //!
  1243. //! <b>Returns</b>: *this
  1244. template <class InputIter>
  1245. basic_string& assign(InputIter first, InputIter last
  1246. #if !defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1247. , typename container_detail::disable_if_convertible<InputIter, size_type>::type * = 0
  1248. #endif
  1249. )
  1250. {
  1251. size_type cur = 0;
  1252. const pointer addr = this->priv_addr();
  1253. CharT *ptr = container_detail::to_raw_pointer(addr);
  1254. const size_type old_size = this->priv_size();
  1255. while (first != last && cur != old_size) {
  1256. Traits::assign(*ptr, *first);
  1257. ++first;
  1258. ++cur;
  1259. ++ptr;
  1260. }
  1261. if (first == last)
  1262. this->erase(addr + cur, addr + old_size);
  1263. else
  1264. this->append(first, last);
  1265. return *this;
  1266. }
  1267. //! <b>Requires</b>: pos <= size().
  1268. //!
  1269. //! <b>Effects</b>: Calls insert(pos, str.data(), str.size()).
  1270. //!
  1271. //! <b>Throws</b>: If memory allocation throws or out_of_range if pos > size().
  1272. //!
  1273. //! <b>Returns</b>: *this
  1274. basic_string& insert(size_type pos, const basic_string& s)
  1275. {
  1276. const size_type sz = this->size();
  1277. if (pos > sz)
  1278. throw_out_of_range("basic_string::insert out of range position");
  1279. if (sz > this->max_size() - s.size())
  1280. throw_length_error("basic_string::insert max_size() exceeded");
  1281. this->insert(this->priv_addr() + pos, s.begin(), s.end());
  1282. return *this;
  1283. }
  1284. //! <b>Requires</b>: pos1 <= size() and pos2 <= str.size()
  1285. //!
  1286. //! <b>Effects</b>: Determines the effective length rlen of the string to insert as
  1287. //! the smaller of n and str.size() - pos2 and calls insert(pos1, str.data() + pos2, rlen).
  1288. //!
  1289. //! <b>Throws</b>: If memory allocation throws or out_of_range if pos1 > size() or pos2 > str.size().
  1290. //!
  1291. //! <b>Returns</b>: *this
  1292. basic_string& insert(size_type pos1, const basic_string& s, size_type pos2, size_type n)
  1293. {
  1294. const size_type sz = this->size();
  1295. const size_type str_size = s.size();
  1296. if (pos1 > sz || pos2 > str_size)
  1297. throw_out_of_range("basic_string::insert out of range position");
  1298. size_type len = container_detail::min_value(n, str_size - pos2);
  1299. if (sz > this->max_size() - len)
  1300. throw_length_error("basic_string::insert max_size() exceeded");
  1301. const CharT *beg_ptr = container_detail::to_raw_pointer(s.begin()) + pos2;
  1302. const CharT *end_ptr = beg_ptr + len;
  1303. this->insert(this->priv_addr() + pos1, beg_ptr, end_ptr);
  1304. return *this;
  1305. }
  1306. //! <b>Requires</b>: s points to an array of at least n elements of CharT and pos <= size().
  1307. //!
  1308. //! <b>Effects</b>: Replaces the string controlled by *this with a string of length size() + n
  1309. //! whose first pos elements are a copy of the initial elements of the original string
  1310. //! controlled by *this and whose next n elements are a copy of the elements in s and whose
  1311. //! remaining elements are a copy of the remaining elements of the original string controlled by *this.
  1312. //!
  1313. //! <b>Throws</b>: If memory allocation throws, out_of_range if pos > size() or
  1314. //! length_error if size() + n > max_size().
  1315. //!
  1316. //! <b>Returns</b>: *this
  1317. basic_string& insert(size_type pos, const CharT* s, size_type n)
  1318. {
  1319. if (pos > this->size())
  1320. throw_out_of_range("basic_string::insert out of range position");
  1321. if (this->size() > this->max_size() - n)
  1322. throw_length_error("basic_string::insert max_size() exceeded");
  1323. this->insert(this->priv_addr() + pos, s, s + n);
  1324. return *this;
  1325. }
  1326. //! <b>Requires</b>: pos <= size() and s points to an array of at least traits::length(s) + 1 elements of CharT
  1327. //!
  1328. //! <b>Effects</b>: Calls insert(pos, s, traits::length(s)).
  1329. //!
  1330. //! <b>Throws</b>: If memory allocation throws, out_of_range if pos > size()
  1331. //! length_error if size() > max_size() - Traits::length(s)
  1332. //!
  1333. //! <b>Returns</b>: *this
  1334. basic_string& insert(size_type pos, const CharT* s)
  1335. {
  1336. if (pos > this->size())
  1337. throw_out_of_range("basic_string::insert out of range position");
  1338. size_type len = Traits::length(s);
  1339. if (this->size() > this->max_size() - len)
  1340. throw_length_error("basic_string::insert max_size() exceeded");
  1341. this->insert(this->priv_addr() + pos, s, s + len);
  1342. return *this;
  1343. }
  1344. //! <b>Effects</b>: Equivalent to insert(pos, basic_string(n, c)).
  1345. //!
  1346. //! <b>Throws</b>: If memory allocation throws, out_of_range if pos > size()
  1347. //! length_error if size() > max_size() - n
  1348. //!
  1349. //! <b>Returns</b>: *this
  1350. basic_string& insert(size_type pos, size_type n, CharT c)
  1351. {
  1352. if (pos > this->size())
  1353. throw_out_of_range("basic_string::insert out of range position");
  1354. if (this->size() > this->max_size() - n)
  1355. throw_length_error("basic_string::insert max_size() exceeded");
  1356. this->insert(const_iterator(this->priv_addr() + pos), n, c);
  1357. return *this;
  1358. }
  1359. //! <b>Requires</b>: p is a valid iterator on *this.
  1360. //!
  1361. //! <b>Effects</b>: inserts a copy of c before the character referred to by p.
  1362. //!
  1363. //! <b>Returns</b>: An iterator which refers to the copy of the inserted character.
  1364. iterator insert(const_iterator p, CharT c)
  1365. {
  1366. size_type new_offset = p - this->priv_addr();
  1367. this->insert(p, cvalue_iterator(c, 1), cvalue_iterator());
  1368. return this->priv_addr() + new_offset;
  1369. }
  1370. //! <b>Requires</b>: p is a valid iterator on *this.
  1371. //!
  1372. //! <b>Effects</b>: Inserts n copies of c before the character referred to by p.
  1373. //!
  1374. //! <b>Returns</b>: an iterator to the first inserted element or p if n is 0.
  1375. iterator insert(const_iterator p, size_type n, CharT c)
  1376. { return this->insert(p, cvalue_iterator(c, n), cvalue_iterator()); }
  1377. //! <b>Requires</b>: p is a valid iterator on *this. [first,last) is a valid range.
  1378. //!
  1379. //! <b>Effects</b>: Equivalent to insert(p - begin(), basic_string(first, last)).
  1380. //!
  1381. //! <b>Returns</b>: an iterator to the first inserted element or p if first == last.
  1382. template <class InputIter>
  1383. iterator insert(const_iterator p, InputIter first, InputIter last
  1384. #if !defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1385. , typename container_detail::disable_if_or
  1386. < void
  1387. , container_detail::is_convertible<InputIter, size_type>
  1388. , container_detail::is_not_input_iterator<InputIter>
  1389. >::type * = 0
  1390. #endif
  1391. )
  1392. {
  1393. const size_type n_pos = p - this->cbegin();
  1394. for ( ; first != last; ++first, ++p) {
  1395. p = this->insert(p, *first);
  1396. }
  1397. return this->begin() + n_pos;
  1398. }
  1399. #if !defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1400. template <class ForwardIter>
  1401. iterator insert(const_iterator p, ForwardIter first, ForwardIter last
  1402. , typename container_detail::disable_if_or
  1403. < void
  1404. , container_detail::is_convertible<ForwardIter, size_type>
  1405. , container_detail::is_input_iterator<ForwardIter>
  1406. >::type * = 0
  1407. )
  1408. {
  1409. const size_type n_pos = p - this->cbegin();
  1410. if (first != last) {
  1411. const size_type n = boost::container::iterator_distance(first, last);
  1412. const size_type old_size = this->priv_size();
  1413. const size_type remaining = this->capacity() - old_size;
  1414. const pointer old_start = this->priv_addr();
  1415. bool enough_capacity = false;
  1416. size_type new_cap = 0;
  1417. //Check if we have enough capacity
  1418. pointer hint = pointer();
  1419. pointer allocation_ret = pointer();
  1420. if (remaining >= n){
  1421. enough_capacity = true;
  1422. }
  1423. else {
  1424. //Otherwise expand current buffer or allocate new storage
  1425. new_cap = this->next_capacity(n);
  1426. hint = old_start;
  1427. allocation_ret = this->allocation_command
  1428. (allocate_new | expand_fwd | expand_bwd, old_size + n + 1, new_cap, hint);
  1429. //Check forward expansion
  1430. if(old_start == allocation_ret){
  1431. enough_capacity = true;
  1432. this->priv_storage(new_cap);
  1433. }
  1434. }
  1435. //Reuse same buffer
  1436. if(enough_capacity){
  1437. const size_type elems_after = old_size - (p - old_start);
  1438. const size_type old_length = old_size;
  1439. if (elems_after >= n) {
  1440. const pointer pointer_past_last = old_start + old_size + 1;
  1441. priv_uninitialized_copy(old_start + (old_size - n + 1),
  1442. pointer_past_last, pointer_past_last);
  1443. this->priv_size(old_size+n);
  1444. Traits::move(const_cast<CharT*>(container_detail::to_raw_pointer(p + n)),
  1445. container_detail::to_raw_pointer(p),
  1446. (elems_after - n) + 1);
  1447. this->priv_copy(first, last, const_cast<CharT*>(container_detail::to_raw_pointer(p)));
  1448. }
  1449. else {
  1450. ForwardIter mid = first;
  1451. boost::container::iterator_advance(mid, elems_after + 1);
  1452. priv_uninitialized_copy(mid, last, old_start + old_size + 1);
  1453. const size_type newer_size = old_size + (n - elems_after);
  1454. this->priv_size(newer_size);
  1455. priv_uninitialized_copy
  1456. (p, const_iterator(old_start + old_length + 1),
  1457. old_start + newer_size);
  1458. this->priv_size(newer_size + elems_after);
  1459. this->priv_copy(first, mid, const_cast<CharT*>(container_detail::to_raw_pointer(p)));
  1460. }
  1461. }
  1462. else{
  1463. pointer new_start = allocation_ret;
  1464. if(!hint){
  1465. //Copy data to new buffer
  1466. size_type new_length = 0;
  1467. //This can't throw, since characters are POD
  1468. new_length += priv_uninitialized_copy
  1469. (const_iterator(old_start), p, new_start);
  1470. new_length += priv_uninitialized_copy
  1471. (first, last, new_start + new_length);
  1472. new_length += priv_uninitialized_copy
  1473. (p, const_iterator(old_start + old_size),
  1474. new_start + new_length);
  1475. this->priv_construct_null(new_start + new_length);
  1476. this->deallocate_block();
  1477. this->is_short(false);
  1478. this->priv_long_addr(new_start);
  1479. this->priv_long_size(new_length);
  1480. this->priv_long_storage(new_cap);
  1481. }
  1482. else{
  1483. //value_type is POD, so backwards expansion is much easier
  1484. //than with vector<T>
  1485. value_type * const oldbuf = container_detail::to_raw_pointer(old_start);
  1486. value_type * const newbuf = container_detail::to_raw_pointer(new_start);
  1487. const value_type *const pos = container_detail::to_raw_pointer(p);
  1488. const size_type before = pos - oldbuf;
  1489. //First move old data
  1490. Traits::move(newbuf, oldbuf, before);
  1491. Traits::move(newbuf + before + n, pos, old_size - before);
  1492. //Now initialize the new data
  1493. priv_uninitialized_copy(first, last, new_start + before);
  1494. this->priv_construct_null(new_start + (old_size + n));
  1495. this->is_short(false);
  1496. this->priv_long_addr(new_start);
  1497. this->priv_long_size(old_size + n);
  1498. this->priv_long_storage(new_cap);
  1499. }
  1500. }
  1501. }
  1502. return this->begin() + n_pos;
  1503. }
  1504. #endif
  1505. //! <b>Effects</b>: Removes the last element from the container.
  1506. //!
  1507. //! <b>Throws</b>: Nothing.
  1508. //!
  1509. //! <b>Complexity</b>: Constant time.
  1510. void pop_back() BOOST_NOEXCEPT_OR_NOTHROW
  1511. {
  1512. BOOST_ASSERT(!this->empty());
  1513. iterator p = this->end();
  1514. this->erase(--p);
  1515. }
  1516. //! <b>Requires</b>: pos <= size()
  1517. //!
  1518. //! <b>Effects</b>: Determines the effective length xlen of the string to be removed as the smaller of n and size() - pos.
  1519. //! The function then replaces the string controlled by *this with a string of length size() - xlen
  1520. //! whose first pos elements are a copy of the initial elements of the original string controlled by *this,
  1521. //! and whose remaining elements are a copy of the elements of the original string controlled by *this
  1522. //! beginning at position pos + xlen.
  1523. //!
  1524. //! <b>Throws</b>: out_of_range if pos > size().
  1525. //!
  1526. //! <b>Returns</b>: *this
  1527. basic_string& erase(size_type pos = 0, size_type n = npos)
  1528. {
  1529. if (pos > this->size())
  1530. throw_out_of_range("basic_string::erase out of range position");
  1531. const pointer addr = this->priv_addr();
  1532. erase(addr + pos, addr + pos + container_detail::min_value(n, this->size() - pos));
  1533. return *this;
  1534. }
  1535. //! <b>Effects</b>: Removes the character referred to by p.
  1536. //!
  1537. //! <b>Throws</b>: Nothing
  1538. //!
  1539. //! <b>Returns</b>: An iterator which points to the element immediately following p prior to the element being
  1540. //! erased. If no such element exists, end() is returned.
  1541. iterator erase(const_iterator p) BOOST_NOEXCEPT_OR_NOTHROW
  1542. {
  1543. // The move includes the terminating null.
  1544. CharT * const ptr = const_cast<CharT*>(container_detail::to_raw_pointer(p));
  1545. const size_type old_size = this->priv_size();
  1546. Traits::move(ptr,
  1547. container_detail::to_raw_pointer(p + 1),
  1548. old_size - (p - this->priv_addr()));
  1549. this->priv_size(old_size-1);
  1550. return iterator(ptr);
  1551. }
  1552. //! <b>Requires</b>: first and last are valid iterators on *this, defining a range [first,last).
  1553. //!
  1554. //! <b>Effects</b>: Removes the characters in the range [first,last).
  1555. //!
  1556. //! <b>Throws</b>: Nothing
  1557. //!
  1558. //! <b>Returns</b>: An iterator which points to the element pointed to by last prior to
  1559. //! the other elements being erased. If no such element exists, end() is returned.
  1560. iterator erase(const_iterator first, const_iterator last) BOOST_NOEXCEPT_OR_NOTHROW
  1561. {
  1562. CharT * f = const_cast<CharT*>(container_detail::to_raw_pointer(first));
  1563. if (first != last) { // The move includes the terminating null.
  1564. const size_type num_erased = last - first;
  1565. const size_type old_size = this->priv_size();
  1566. Traits::move(f,
  1567. container_detail::to_raw_pointer(last),
  1568. (old_size + 1)-(last - this->priv_addr()));
  1569. const size_type new_length = old_size - num_erased;
  1570. this->priv_size(new_length);
  1571. }
  1572. return iterator(f);
  1573. }
  1574. //! <b>Effects</b>: Erases all the elements of the vector.
  1575. //!
  1576. //! <b>Throws</b>: Nothing.
  1577. //!
  1578. //! <b>Complexity</b>: Linear to the number of elements in the vector.
  1579. void clear() BOOST_NOEXCEPT_OR_NOTHROW
  1580. {
  1581. if (!this->empty()) {
  1582. Traits::assign(*this->priv_addr(), CharT(0));
  1583. this->priv_size(0);
  1584. }
  1585. }
  1586. //! <b>Requires</b>: pos1 <= size().
  1587. //!
  1588. //! <b>Effects</b>: Calls replace(pos1, n1, str.data(), str.size()).
  1589. //!
  1590. //! <b>Throws</b>: if memory allocation throws or out_of_range if pos1 > size().
  1591. //!
  1592. //! <b>Returns</b>: *this
  1593. basic_string& replace(size_type pos1, size_type n1, const basic_string& str)
  1594. {
  1595. if (pos1 > this->size())
  1596. throw_out_of_range("basic_string::replace out of range position");
  1597. const size_type len = container_detail::min_value(n1, this->size() - pos1);
  1598. if (this->size() - len >= this->max_size() - str.size())
  1599. throw_length_error("basic_string::replace max_size() exceeded");
  1600. const pointer addr = this->priv_addr();
  1601. return this->replace( const_iterator(addr + pos1)
  1602. , const_iterator(addr + pos1 + len)
  1603. , str.begin(), str.end());
  1604. }
  1605. //! <b>Requires</b>: pos1 <= size() and pos2 <= str.size().
  1606. //!
  1607. //! <b>Effects</b>: Determines the effective length rlen of the string to be
  1608. //! inserted as the smaller of n2 and str.size() - pos2 and calls
  1609. //! replace(pos1, n1, str.data() + pos2, rlen).
  1610. //!
  1611. //! <b>Throws</b>: if memory allocation throws, out_of_range if pos1 > size() or pos2 > str.size().
  1612. //!
  1613. //! <b>Returns</b>: *this
  1614. basic_string& replace(size_type pos1, size_type n1,
  1615. const basic_string& str, size_type pos2, size_type n2)
  1616. {
  1617. if (pos1 > this->size() || pos2 > str.size())
  1618. throw_out_of_range("basic_string::replace out of range position");
  1619. const size_type len1 = container_detail::min_value(n1, this->size() - pos1);
  1620. const size_type len2 = container_detail::min_value(n2, str.size() - pos2);
  1621. if (this->size() - len1 >= this->max_size() - len2)
  1622. throw_length_error("basic_string::replace max_size() exceeded");
  1623. const pointer addr = this->priv_addr();
  1624. const pointer straddr = str.priv_addr();
  1625. return this->replace(addr + pos1, addr + pos1 + len1,
  1626. straddr + pos2, straddr + pos2 + len2);
  1627. }
  1628. //! <b>Requires</b>: pos1 <= size() and s points to an array of at least n2 elements of CharT.
  1629. //!
  1630. //! <b>Effects</b>: Determines the effective length xlen of the string to be removed as the
  1631. //! smaller of n1 and size() - pos1. If size() - xlen >= max_size() - n2 throws length_error.
  1632. //! Otherwise, the function replaces the string controlled by *this with a string of
  1633. //! length size() - xlen + n2 whose first pos1 elements are a copy of the initial elements
  1634. //! of the original string controlled by *this, whose next n2 elements are a copy of the
  1635. //! initial n2 elements of s, and whose remaining elements are a copy of the elements of
  1636. //! the original string controlled by *this beginning at position pos + xlen.
  1637. //!
  1638. //! <b>Throws</b>: if memory allocation throws, out_of_range if pos1 > size() or length_error
  1639. //! if the length of the resulting string would exceed max_size()
  1640. //!
  1641. //! <b>Returns</b>: *this
  1642. basic_string& replace(size_type pos1, size_type n1, const CharT* s, size_type n2)
  1643. {
  1644. if (pos1 > this->size())
  1645. throw_out_of_range("basic_string::replace out of range position");
  1646. const size_type len = container_detail::min_value(n1, this->size() - pos1);
  1647. if (n2 > this->max_size() || size() - len >= this->max_size() - n2)
  1648. throw_length_error("basic_string::replace max_size() exceeded");
  1649. const pointer addr = this->priv_addr();
  1650. return this->replace(addr + pos1, addr + pos1 + len, s, s + n2);
  1651. }
  1652. //! <b>Requires</b>: pos1 <= size() and s points to an array of at least n2 elements of CharT.
  1653. //!
  1654. //! <b>Effects</b>: Determines the effective length xlen of the string to be removed as the smaller
  1655. //! of n1 and size() - pos1. If size() - xlen >= max_size() - n2 throws length_error. Otherwise,
  1656. //! the function replaces the string controlled by *this with a string of length size() - xlen + n2
  1657. //! whose first pos1 elements are a copy of the initial elements of the original string controlled
  1658. //! by *this, whose next n2 elements are a copy of the initial n2 elements of s, and whose
  1659. //! remaining elements are a copy of the elements of the original string controlled by *this
  1660. //! beginning at position pos + xlen.
  1661. //!
  1662. //! <b>Throws</b>: if memory allocation throws, out_of_range if pos1 > size() or length_error
  1663. //! if the length of the resulting string would exceed max_size()
  1664. //!
  1665. //! <b>Returns</b>: *this
  1666. basic_string& replace(size_type pos, size_type n1, const CharT* s)
  1667. {
  1668. if (pos > this->size())
  1669. throw_out_of_range("basic_string::replace out of range position");
  1670. const size_type len = container_detail::min_value(n1, this->size() - pos);
  1671. const size_type n2 = Traits::length(s);
  1672. if (n2 > this->max_size() || this->size() - len >= this->max_size() - n2)
  1673. throw_length_error("basic_string::replace max_size() exceeded");
  1674. const pointer addr = this->priv_addr();
  1675. return this->replace(addr + pos, addr + pos + len,
  1676. s, s + Traits::length(s));
  1677. }
  1678. //! <b>Requires</b>: pos1 <= size().
  1679. //!
  1680. //! <b>Effects</b>: Equivalent to replace(pos1, n1, basic_string(n2, c)).
  1681. //!
  1682. //! <b>Throws</b>: if memory allocation throws, out_of_range if pos1 > size() or length_error
  1683. //! if the length of the resulting string would exceed max_size()
  1684. //!
  1685. //! <b>Returns</b>: *this
  1686. basic_string& replace(size_type pos1, size_type n1, size_type n2, CharT c)
  1687. {
  1688. if (pos1 > this->size())
  1689. throw_out_of_range("basic_string::replace out of range position");
  1690. const size_type len = container_detail::min_value(n1, this->size() - pos1);
  1691. if (n2 > this->max_size() || this->size() - len >= this->max_size() - n2)
  1692. throw_length_error("basic_string::replace max_size() exceeded");
  1693. const pointer addr = this->priv_addr();
  1694. return this->replace(addr + pos1, addr + pos1 + len, n2, c);
  1695. }
  1696. //! <b>Requires</b>: [begin(),i1) and [i1,i2) are valid ranges.
  1697. //!
  1698. //! <b>Effects</b>: Calls replace(i1 - begin(), i2 - i1, str).
  1699. //!
  1700. //! <b>Throws</b>: if memory allocation throws
  1701. //!
  1702. //! <b>Returns</b>: *this
  1703. basic_string& replace(const_iterator i1, const_iterator i2, const basic_string& str)
  1704. { return this->replace(i1, i2, str.begin(), str.end()); }
  1705. //! <b>Requires</b>: [begin(),i1) and [i1,i2) are valid ranges and
  1706. //! s points to an array of at least n elements
  1707. //!
  1708. //! <b>Effects</b>: Calls replace(i1 - begin(), i2 - i1, s, n).
  1709. //!
  1710. //! <b>Throws</b>: if memory allocation throws
  1711. //!
  1712. //! <b>Returns</b>: *this
  1713. basic_string& replace(const_iterator i1, const_iterator i2, const CharT* s, size_type n)
  1714. { return this->replace(i1, i2, s, s + n); }
  1715. //! <b>Requires</b>: [begin(),i1) and [i1,i2) are valid ranges and s points to an
  1716. //! array of at least traits::length(s) + 1 elements of CharT.
  1717. //!
  1718. //! <b>Effects</b>: Calls replace(i1 - begin(), i2 - i1, s, traits::length(s)).
  1719. //!
  1720. //! <b>Throws</b>: if memory allocation throws
  1721. //!
  1722. //! <b>Returns</b>: *this
  1723. basic_string& replace(const_iterator i1, const_iterator i2, const CharT* s)
  1724. { return this->replace(i1, i2, s, s + Traits::length(s)); }
  1725. //! <b>Requires</b>: [begin(),i1) and [i1,i2) are valid ranges.
  1726. //!
  1727. //! <b>Effects</b>: Calls replace(i1 - begin(), i2 - i1, basic_string(n, c)).
  1728. //!
  1729. //! <b>Throws</b>: if memory allocation throws
  1730. //!
  1731. //! <b>Returns</b>: *this
  1732. basic_string& replace(const_iterator i1, const_iterator i2, size_type n, CharT c)
  1733. {
  1734. const size_type len = static_cast<size_type>(i2 - i1);
  1735. if (len >= n) {
  1736. Traits::assign(const_cast<CharT*>(container_detail::to_raw_pointer(i1)), n, c);
  1737. erase(i1 + n, i2);
  1738. }
  1739. else {
  1740. Traits::assign(const_cast<CharT*>(container_detail::to_raw_pointer(i1)), len, c);
  1741. insert(i2, n - len, c);
  1742. }
  1743. return *this;
  1744. }
  1745. //! <b>Requires</b>: [begin(),i1), [i1,i2) and [j1,j2) are valid ranges.
  1746. //!
  1747. //! <b>Effects</b>: Calls replace(i1 - begin(), i2 - i1, basic_string(j1, j2)).
  1748. //!
  1749. //! <b>Throws</b>: if memory allocation throws
  1750. //!
  1751. //! <b>Returns</b>: *this
  1752. template <class InputIter>
  1753. basic_string& replace(const_iterator i1, const_iterator i2, InputIter j1, InputIter j2
  1754. #if !defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1755. , typename container_detail::disable_if_or
  1756. < void
  1757. , container_detail::is_convertible<InputIter, size_type>
  1758. , container_detail::is_input_iterator<InputIter>
  1759. >::type * = 0
  1760. #endif
  1761. )
  1762. {
  1763. for ( ; i1 != i2 && j1 != j2; ++i1, ++j1){
  1764. Traits::assign(*const_cast<CharT*>(container_detail::to_raw_pointer(i1)), *j1);
  1765. }
  1766. if (j1 == j2)
  1767. this->erase(i1, i2);
  1768. else
  1769. this->insert(i2, j1, j2);
  1770. return *this;
  1771. }
  1772. #if !defined(BOOST_CONTAINER_DOXYGEN_INVOKED)
  1773. template <class ForwardIter>
  1774. basic_string& replace(const_iterator i1, const_iterator i2, ForwardIter j1, ForwardIter j2
  1775. , typename container_detail::disable_if_or
  1776. < void
  1777. , container_detail::is_convertible<ForwardIter, size_type>
  1778. , container_detail::is_not_input_iterator<ForwardIter>
  1779. >::type * = 0
  1780. )
  1781. {
  1782. difference_type n = boost::container::iterator_distance(j1, j2);
  1783. const difference_type len = i2 - i1;
  1784. if (len >= n) {
  1785. this->priv_copy(j1, j2, const_cast<CharT*>(container_detail::to_raw_pointer(i1)));
  1786. this->erase(i1 + n, i2);
  1787. }
  1788. else {
  1789. ForwardIter m = j1;
  1790. boost::container::iterator_advance(m, len);
  1791. this->priv_copy(j1, m, const_cast<CharT*>(container_detail::to_raw_pointer(i1)));
  1792. this->insert(i2, m, j2);
  1793. }
  1794. return *this;
  1795. }
  1796. #endif
  1797. //! <b>Requires</b>: pos <= size()
  1798. //!
  1799. //! <b>Effects</b>: Determines the effective length rlen of the string to copy as the
  1800. //! smaller of n and size() - pos. s shall designate an array of at least rlen elements.
  1801. //! The function then replaces the string designated by s with a string of length rlen
  1802. //! whose elements are a copy of the string controlled by *this beginning at position pos.
  1803. //! The function does not append a null object to the string designated by s.
  1804. //!
  1805. //! <b>Throws</b>: if memory allocation throws, out_of_range if pos > size().
  1806. //!
  1807. //! <b>Returns</b>: rlen
  1808. size_type copy(CharT* s, size_type n, size_type pos = 0) const
  1809. {
  1810. if (pos > this->size())
  1811. throw_out_of_range("basic_string::copy out of range position");
  1812. const size_type len = container_detail::min_value(n, this->size() - pos);
  1813. Traits::copy(s, container_detail::to_raw_pointer(this->priv_addr() + pos), len);
  1814. return len;
  1815. }
  1816. //! <b>Effects</b>: *this contains the same sequence of characters that was in s,
  1817. //! s contains the same sequence of characters that was in *this.
  1818. //!
  1819. //! <b>Throws</b>: Nothing
  1820. void swap(basic_string& x)
  1821. BOOST_NOEXCEPT_IF(allocator_traits_type::propagate_on_container_swap::value
  1822. || allocator_traits_type::is_always_equal::value)
  1823. {
  1824. this->base_t::swap_data(x);
  1825. container_detail::bool_<allocator_traits_type::propagate_on_container_swap::value> flag;
  1826. container_detail::swap_alloc(this->alloc(), x.alloc(), flag);
  1827. }
  1828. //////////////////////////////////////////////
  1829. //
  1830. // data access
  1831. //
  1832. //////////////////////////////////////////////
  1833. //! <b>Requires</b>: The program shall not alter any of the values stored in the character array.
  1834. //!
  1835. //! <b>Returns</b>: A pointer p such that p + i == &operator[](i) for each i in [0,size()].
  1836. //!
  1837. //! <b>Complexity</b>: constant time.
  1838. const CharT* c_str() const BOOST_NOEXCEPT_OR_NOTHROW
  1839. { return container_detail::to_raw_pointer(this->priv_addr()); }
  1840. //! <b>Requires</b>: The program shall not alter any of the values stored in the character array.
  1841. //!
  1842. //! <b>Returns</b>: A pointer p such that p + i == &operator[](i) for each i in [0,size()].
  1843. //!
  1844. //! <b>Complexity</b>: constant time.
  1845. const CharT* data() const BOOST_NOEXCEPT_OR_NOTHROW
  1846. { return container_detail::to_raw_pointer(this->priv_addr()); }
  1847. //////////////////////////////////////////////
  1848. //
  1849. // string operations
  1850. //
  1851. //////////////////////////////////////////////
  1852. //! <b>Effects</b>: Determines the lowest position xpos, if possible, such that both
  1853. //! of the following conditions obtain: 19 pos <= xpos and xpos + str.size() <= size();
  1854. //! 2) traits::eq(at(xpos+I), str.at(I)) for all elements I of the string controlled by str.
  1855. //!
  1856. //! <b>Throws</b>: Nothing
  1857. //!
  1858. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  1859. size_type find(const basic_string& s, size_type pos = 0) const
  1860. { return find(s.c_str(), pos, s.size()); }
  1861. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  1862. //!
  1863. //! <b>Throws</b>: Nothing
  1864. //!
  1865. //! <b>Returns</b>: find(basic_string<CharT,traits,Allocator>(s,n),pos).
  1866. size_type find(const CharT* s, size_type pos, size_type n) const
  1867. {
  1868. if (pos + n > this->size())
  1869. return npos;
  1870. else {
  1871. const pointer addr = this->priv_addr();
  1872. pointer finish = addr + this->priv_size();
  1873. const const_iterator result =
  1874. std::search(container_detail::to_raw_pointer(addr + pos),
  1875. container_detail::to_raw_pointer(finish),
  1876. s, s + n, Eq_traits<Traits>());
  1877. return result != finish ? result - begin() : npos;
  1878. }
  1879. }
  1880. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  1881. //!
  1882. //! <b>Throws</b>: Nothing
  1883. //!
  1884. //! <b>Returns</b>: find(basic_string(s), pos).
  1885. size_type find(const CharT* s, size_type pos = 0) const
  1886. { return this->find(s, pos, Traits::length(s)); }
  1887. //! <b>Throws</b>: Nothing
  1888. //!
  1889. //! <b>Returns</b>: find(basic_string<CharT,traits,Allocator>(1,c), pos).
  1890. size_type find(CharT c, size_type pos = 0) const
  1891. {
  1892. const size_type sz = this->size();
  1893. if (pos >= sz)
  1894. return npos;
  1895. else {
  1896. const pointer addr = this->priv_addr();
  1897. pointer finish = addr + sz;
  1898. const const_iterator result =
  1899. std::find_if(addr + pos, finish,
  1900. std::bind2nd(Eq_traits<Traits>(), c));
  1901. return result != finish ? result - begin() : npos;
  1902. }
  1903. }
  1904. //! <b>Effects</b>: Determines the highest position xpos, if possible, such
  1905. //! that both of the following conditions obtain:
  1906. //! a) xpos <= pos and xpos + str.size() <= size();
  1907. //! b) traits::eq(at(xpos+I), str.at(I)) for all elements I of the string controlled by str.
  1908. //!
  1909. //! <b>Throws</b>: Nothing
  1910. //!
  1911. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  1912. size_type rfind(const basic_string& str, size_type pos = npos) const
  1913. { return rfind(str.c_str(), pos, str.size()); }
  1914. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  1915. //!
  1916. //! <b>Throws</b>: Nothing
  1917. //!
  1918. //! <b>Returns</b>: rfind(basic_string(s, n), pos).
  1919. size_type rfind(const CharT* s, size_type pos, size_type n) const
  1920. {
  1921. const size_type len = this->size();
  1922. if (n > len)
  1923. return npos;
  1924. else if (n == 0)
  1925. return container_detail::min_value(len, pos);
  1926. else {
  1927. const const_iterator last = begin() + container_detail::min_value(len - n, pos) + n;
  1928. const const_iterator result = find_end(begin(), last,
  1929. s, s + n,
  1930. Eq_traits<Traits>());
  1931. return result != last ? result - begin() : npos;
  1932. }
  1933. }
  1934. //! <b>Requires</b>: pos <= size() and s points to an array of at least
  1935. //! traits::length(s) + 1 elements of CharT.
  1936. //!
  1937. //! <b>Throws</b>: Nothing
  1938. //!
  1939. //! <b>Returns</b>: rfind(basic_string(s), pos).
  1940. size_type rfind(const CharT* s, size_type pos = npos) const
  1941. { return rfind(s, pos, Traits::length(s)); }
  1942. //! <b>Throws</b>: Nothing
  1943. //!
  1944. //! <b>Returns</b>: rfind(basic_string<CharT,traits,Allocator>(1,c),pos).
  1945. size_type rfind(CharT c, size_type pos = npos) const
  1946. {
  1947. const size_type len = this->size();
  1948. if (len < 1)
  1949. return npos;
  1950. else {
  1951. const const_iterator last = begin() + container_detail::min_value(len - 1, pos) + 1;
  1952. const_reverse_iterator rresult =
  1953. std::find_if(const_reverse_iterator(last), rend(),
  1954. std::bind2nd(Eq_traits<Traits>(), c));
  1955. return rresult != rend() ? (rresult.base() - 1) - begin() : npos;
  1956. }
  1957. }
  1958. //! <b>Effects</b>: Determines the lowest position xpos, if possible, such that both of the
  1959. //! following conditions obtain: a) pos <= xpos and xpos < size();
  1960. //! b) traits::eq(at(xpos), str.at(I)) for some element I of the string controlled by str.
  1961. //!
  1962. //! <b>Throws</b>: Nothing
  1963. //!
  1964. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  1965. size_type find_first_of(const basic_string& s, size_type pos = 0) const
  1966. { return find_first_of(s.c_str(), pos, s.size()); }
  1967. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  1968. //!
  1969. //! <b>Throws</b>: Nothing
  1970. //!
  1971. //! <b>Returns</b>: find_first_of(basic_string(s, n), pos).
  1972. size_type find_first_of(const CharT* s, size_type pos, size_type n) const
  1973. {
  1974. const size_type sz = this->size();
  1975. if (pos >= sz)
  1976. return npos;
  1977. else {
  1978. const pointer addr = this->priv_addr();
  1979. pointer finish = addr + sz;
  1980. const_iterator result = std::find_first_of
  1981. (addr + pos, finish, s, s + n, Eq_traits<Traits>());
  1982. return result != finish ? result - this->begin() : npos;
  1983. }
  1984. }
  1985. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  1986. //!
  1987. //! <b>Throws</b>: Nothing
  1988. //!
  1989. //! <b>Returns</b>: find_first_of(basic_string(s), pos).
  1990. size_type find_first_of(const CharT* s, size_type pos = 0) const
  1991. { return find_first_of(s, pos, Traits::length(s)); }
  1992. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  1993. //!
  1994. //! <b>Throws</b>: Nothing
  1995. //!
  1996. //! <b>Returns</b>: find_first_of(basic_string<CharT,traits,Allocator>(1,c), pos).
  1997. size_type find_first_of(CharT c, size_type pos = 0) const
  1998. { return find(c, pos); }
  1999. //! <b>Effects</b>: Determines the highest position xpos, if possible, such that both of
  2000. //! the following conditions obtain: a) xpos <= pos and xpos < size(); b)
  2001. //! traits::eq(at(xpos), str.at(I)) for some element I of the string controlled by str.
  2002. //!
  2003. //! <b>Throws</b>: Nothing
  2004. //!
  2005. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  2006. size_type find_last_of(const basic_string& str, size_type pos = npos) const
  2007. { return find_last_of(str.c_str(), pos, str.size()); }
  2008. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  2009. //!
  2010. //! <b>Throws</b>: Nothing
  2011. //!
  2012. //! <b>Returns</b>: find_last_of(basic_string(s, n), pos).
  2013. size_type find_last_of(const CharT* s, size_type pos, size_type n) const
  2014. {
  2015. const size_type len = this->size();
  2016. if (len < 1)
  2017. return npos;
  2018. else {
  2019. const pointer addr = this->priv_addr();
  2020. const const_iterator last = addr + container_detail::min_value(len - 1, pos) + 1;
  2021. const const_reverse_iterator rresult =
  2022. std::find_first_of(const_reverse_iterator(last), rend(),
  2023. s, s + n, Eq_traits<Traits>());
  2024. return rresult != rend() ? (rresult.base() - 1) - addr : npos;
  2025. }
  2026. }
  2027. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  2028. //!
  2029. //! <b>Throws</b>: Nothing
  2030. //!
  2031. //! <b>Returns</b>: find_last_of(basic_string<CharT,traits,Allocator>(1,c),pos).
  2032. size_type find_last_of(const CharT* s, size_type pos = npos) const
  2033. { return find_last_of(s, pos, Traits::length(s)); }
  2034. //! <b>Throws</b>: Nothing
  2035. //!
  2036. //! <b>Returns</b>: find_last_of(basic_string(s), pos).
  2037. size_type find_last_of(CharT c, size_type pos = npos) const
  2038. { return rfind(c, pos); }
  2039. //! <b>Effects</b>: Determines the lowest position xpos, if possible, such that
  2040. //! both of the following conditions obtain:
  2041. //! a) pos <= xpos and xpos < size(); b) traits::eq(at(xpos), str.at(I)) for no
  2042. //! element I of the string controlled by str.
  2043. //!
  2044. //! <b>Throws</b>: Nothing
  2045. //!
  2046. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  2047. size_type find_first_not_of(const basic_string& str, size_type pos = 0) const
  2048. { return find_first_not_of(str.c_str(), pos, str.size()); }
  2049. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  2050. //!
  2051. //! <b>Throws</b>: Nothing
  2052. //!
  2053. //! <b>Returns</b>: find_first_not_of(basic_string(s, n), pos).
  2054. size_type find_first_not_of(const CharT* s, size_type pos, size_type n) const
  2055. {
  2056. if (pos > this->size())
  2057. return npos;
  2058. else {
  2059. const pointer addr = this->priv_addr();
  2060. const pointer finish = addr + this->priv_size();
  2061. const const_iterator result = std::find_if
  2062. (addr + pos, finish, Not_within_traits<Traits>(s, s + n));
  2063. return result != finish ? result - addr : npos;
  2064. }
  2065. }
  2066. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  2067. //!
  2068. //! <b>Throws</b>: Nothing
  2069. //!
  2070. //! <b>Returns</b>: find_first_not_of(basic_string(s), pos).
  2071. size_type find_first_not_of(const CharT* s, size_type pos = 0) const
  2072. { return find_first_not_of(s, pos, Traits::length(s)); }
  2073. //! <b>Throws</b>: Nothing
  2074. //!
  2075. //! <b>Returns</b>: find_first_not_of(basic_string(1, c), pos).
  2076. size_type find_first_not_of(CharT c, size_type pos = 0) const
  2077. {
  2078. if (pos > this->size())
  2079. return npos;
  2080. else {
  2081. const pointer addr = this->priv_addr();
  2082. const pointer finish = addr + this->priv_size();
  2083. const const_iterator result
  2084. = std::find_if(addr + pos, finish,
  2085. std::not1(std::bind2nd(Eq_traits<Traits>(), c)));
  2086. return result != finish ? result - begin() : npos;
  2087. }
  2088. }
  2089. //! <b>Effects</b>: Determines the highest position xpos, if possible, such that
  2090. //! both of the following conditions obtain: a) xpos <= pos and xpos < size();
  2091. //! b) traits::eq(at(xpos), str.at(I)) for no element I of the string controlled by str.
  2092. //!
  2093. //! <b>Throws</b>: Nothing
  2094. //!
  2095. //! <b>Returns</b>: xpos if the function can determine such a value for xpos. Otherwise, returns npos.
  2096. size_type find_last_not_of(const basic_string& str, size_type pos = npos) const
  2097. { return find_last_not_of(str.c_str(), pos, str.size()); }
  2098. //! <b>Requires</b>: s points to an array of at least n elements of CharT.
  2099. //!
  2100. //! <b>Throws</b>: Nothing
  2101. //!
  2102. //! <b>Returns</b>: find_last_not_of(basic_string(s, n), pos).
  2103. size_type find_last_not_of(const CharT* s, size_type pos, size_type n) const
  2104. {
  2105. const size_type len = this->size();
  2106. if (len < 1)
  2107. return npos;
  2108. else {
  2109. const const_iterator last = begin() + container_detail::min_value(len - 1, pos) + 1;
  2110. const const_reverse_iterator rresult =
  2111. std::find_if(const_reverse_iterator(last), rend(),
  2112. Not_within_traits<Traits>(s, s + n));
  2113. return rresult != rend() ? (rresult.base() - 1) - begin() : npos;
  2114. }
  2115. }
  2116. //! <b>Requires</b>: s points to an array of at least traits::length(s) + 1 elements of CharT.
  2117. //!
  2118. //! <b>Throws</b>: Nothing
  2119. //!
  2120. //! <b>Returns</b>: find_last_not_of(basic_string(s), pos).
  2121. size_type find_last_not_of(const CharT* s, size_type pos = npos) const
  2122. { return find_last_not_of(s, pos, Traits::length(s)); }
  2123. //! <b>Throws</b>: Nothing
  2124. //!
  2125. //! <b>Returns</b>: find_last_not_of(basic_string(1, c), pos).
  2126. size_type find_last_not_of(CharT c, size_type pos = npos) const
  2127. {
  2128. const size_type len = this->size();
  2129. if (len < 1)
  2130. return npos;
  2131. else {
  2132. const const_iterator last = begin() + container_detail::min_value(len - 1, pos) + 1;
  2133. const const_reverse_iterator rresult =
  2134. std::find_if(const_reverse_iterator(last), rend(),
  2135. std::not1(std::bind2nd(Eq_traits<Traits>(), c)));
  2136. return rresult != rend() ? (rresult.base() - 1) - begin() : npos;
  2137. }
  2138. }
  2139. //! <b>Requires</b>: Requires: pos <= size()
  2140. //!
  2141. //! <b>Effects</b>: Determines the effective length rlen of the string to copy as
  2142. //! the smaller of n and size() - pos.
  2143. //!
  2144. //! <b>Throws</b>: If memory allocation throws or out_of_range if pos > size().
  2145. //!
  2146. //! <b>Returns</b>: basic_string<CharT,traits,Allocator>(data()+pos,rlen).
  2147. basic_string substr(size_type pos = 0, size_type n = npos) const
  2148. {
  2149. if (pos > this->size())
  2150. throw_out_of_range("basic_string::substr out of range position");
  2151. const pointer addr = this->priv_addr();
  2152. return basic_string(addr + pos,
  2153. addr + pos + container_detail::min_value(n, size() - pos), this->alloc());
  2154. }
  2155. //! <b>Effects</b>: Determines the effective length rlen of the string to copy as
  2156. //! the smaller of size() and str.size(). The function then compares the two strings by
  2157. //! calling traits::compare(data(), str.data(), rlen).
  2158. //!
  2159. //! <b>Throws</b>: Nothing
  2160. //!
  2161. //! <b>Returns</b>: The nonzero result if the result of the comparison is nonzero.
  2162. //! Otherwise, returns a value < 0 if size() < str.size(), a 0 value if size() == str.size(),
  2163. //! and value > 0 if size() > str.size()
  2164. int compare(const basic_string& str) const
  2165. {
  2166. const pointer addr = this->priv_addr();
  2167. const pointer str_addr = str.priv_addr();
  2168. return s_compare(addr, addr + this->priv_size(), str_addr, str_addr + str.priv_size());
  2169. }
  2170. //! <b>Requires</b>: pos1 <= size()
  2171. //!
  2172. //! <b>Effects</b>: Determines the effective length rlen of the string to copy as
  2173. //! the smaller of
  2174. //!
  2175. //! <b>Throws</b>: out_of_range if pos1 > size()
  2176. //!
  2177. //! <b>Returns</b>:basic_string(*this,pos1,n1).compare(str).
  2178. int compare(size_type pos1, size_type n1, const basic_string& str) const
  2179. {
  2180. if (pos1 > this->size())
  2181. throw_out_of_range("basic_string::compare out of range position");
  2182. const pointer addr = this->priv_addr();
  2183. const pointer str_addr = str.priv_addr();
  2184. return s_compare(addr + pos1,
  2185. addr + pos1 + container_detail::min_value(n1, this->size() - pos1),
  2186. str_addr, str_addr + str.priv_size());
  2187. }
  2188. //! <b>Requires</b>: pos1 <= size() and pos2 <= str.size()
  2189. //!
  2190. //! <b>Effects</b>: Determines the effective length rlen of the string to copy as
  2191. //! the smaller of
  2192. //!
  2193. //! <b>Throws</b>: out_of_range if pos1 > size() or pos2 > str.size()
  2194. //!
  2195. //! <b>Returns</b>: basic_string(*this, pos1, n1).compare(basic_string(str, pos2, n2)).
  2196. int compare(size_type pos1, size_type n1, const basic_string& str, size_type pos2, size_type n2) const
  2197. {
  2198. if (pos1 > this->size() || pos2 > str.size())
  2199. throw_out_of_range("basic_string::compare out of range position");
  2200. const pointer addr = this->priv_addr();
  2201. const pointer str_addr = str.priv_addr();
  2202. return s_compare(addr + pos1,
  2203. addr + pos1 + container_detail::min_value(n1, this->size() - pos1),
  2204. str_addr + pos2,
  2205. str_addr + pos2 + container_detail::min_value(n2, str.size() - pos2));
  2206. }
  2207. //! <b>Throws</b>: Nothing
  2208. //!
  2209. //! <b>Returns</b>: compare(basic_string(s)).
  2210. int compare(const CharT* s) const
  2211. {
  2212. const pointer addr = this->priv_addr();
  2213. return s_compare(addr, addr + this->priv_size(), s, s + Traits::length(s));
  2214. }
  2215. //! <b>Requires</b>: pos1 > size() and s points to an array of at least n2 elements of CharT.
  2216. //!
  2217. //! <b>Throws</b>: out_of_range if pos1 > size()
  2218. //!
  2219. //! <b>Returns</b>: basic_string(*this, pos, n1).compare(basic_string(s, n2)).
  2220. int compare(size_type pos1, size_type n1, const CharT* s, size_type n2) const
  2221. {
  2222. if (pos1 > this->size())
  2223. throw_out_of_range("basic_string::compare out of range position");
  2224. const pointer addr = this->priv_addr();
  2225. return s_compare( addr + pos1,
  2226. addr + pos1 + container_detail::min_value(n1, this->size() - pos1),
  2227. s, s + n2);
  2228. }
  2229. //! <b>Requires</b>: pos1 > size() and s points to an array of at least traits::length(s) + 1 elements of CharT.
  2230. //!
  2231. //! <b>Throws</b>: out_of_range if pos1 > size()
  2232. //!
  2233. //! <b>Returns</b>: basic_string(*this, pos, n1).compare(basic_string(s, n2)).
  2234. int compare(size_type pos1, size_type n1, const CharT* s) const
  2235. { return this->compare(pos1, n1, s, Traits::length(s)); }
  2236. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2237. private:
  2238. void priv_reserve(size_type res_arg, const bool null_terminate = true)
  2239. {
  2240. if (res_arg > this->max_size()){
  2241. throw_length_error("basic_string::reserve max_size() exceeded");
  2242. }
  2243. if (this->capacity() < res_arg){
  2244. size_type n = container_detail::max_value(res_arg, this->size()) + 1;
  2245. size_type new_cap = this->next_capacity(n);
  2246. pointer reuse = 0;
  2247. pointer new_start = this->allocation_command(allocate_new, n, new_cap, reuse);
  2248. size_type new_length = 0;
  2249. const pointer addr = this->priv_addr();
  2250. new_length += priv_uninitialized_copy
  2251. (addr, addr + this->priv_size(), new_start);
  2252. if(null_terminate){
  2253. this->priv_construct_null(new_start + new_length);
  2254. }
  2255. this->deallocate_block();
  2256. this->is_short(false);
  2257. this->priv_long_addr(new_start);
  2258. this->priv_long_size(new_length);
  2259. this->priv_storage(new_cap);
  2260. }
  2261. }
  2262. static int s_compare(const_pointer f1, const_pointer l1,
  2263. const_pointer f2, const_pointer l2)
  2264. {
  2265. const difference_type n1 = l1 - f1;
  2266. const difference_type n2 = l2 - f2;
  2267. const int cmp = Traits::compare(container_detail::to_raw_pointer(f1),
  2268. container_detail::to_raw_pointer(f2),
  2269. container_detail::min_value(n1, n2));
  2270. return cmp != 0 ? cmp : (n1 < n2 ? -1 : (n1 > n2 ? 1 : 0));
  2271. }
  2272. template<class AllocVersion>
  2273. void priv_shrink_to_fit_dynamic_buffer
  2274. ( AllocVersion
  2275. , typename container_detail::enable_if<container_detail::is_same<AllocVersion, version_1> >::type* = 0)
  2276. {
  2277. //Allocate a new buffer.
  2278. size_type real_cap = 0;
  2279. const pointer long_addr = this->priv_long_addr();
  2280. const size_type long_size = this->priv_long_size();
  2281. const size_type long_storage = this->priv_long_storage();
  2282. //We can make this nothrow as chars are always NoThrowCopyables
  2283. BOOST_TRY{
  2284. pointer reuse = 0;
  2285. real_cap = long_size+1;
  2286. const pointer ret = this->allocation_command(allocate_new, long_size+1, real_cap, reuse);
  2287. //Copy and update
  2288. Traits::copy( container_detail::to_raw_pointer(ret)
  2289. , container_detail::to_raw_pointer(this->priv_long_addr())
  2290. , long_size+1);
  2291. this->priv_long_addr(ret);
  2292. this->priv_storage(real_cap);
  2293. //And release old buffer
  2294. this->alloc().deallocate(long_addr, long_storage);
  2295. }
  2296. BOOST_CATCH(...){
  2297. return;
  2298. }
  2299. BOOST_CATCH_END
  2300. }
  2301. template<class AllocVersion>
  2302. void priv_shrink_to_fit_dynamic_buffer
  2303. ( AllocVersion
  2304. , typename container_detail::enable_if<container_detail::is_same<AllocVersion, version_2> >::type* = 0)
  2305. {
  2306. size_type received_size = this->priv_long_size()+1;
  2307. pointer hint = this->priv_long_addr();
  2308. if(this->alloc().allocation_command
  2309. ( shrink_in_place | nothrow_allocation, this->priv_long_storage(), received_size, hint)){
  2310. this->priv_storage(received_size);
  2311. }
  2312. }
  2313. void priv_construct_null(pointer p)
  2314. { this->construct(p, CharT(0)); }
  2315. // Helper functions used by constructors. It is a severe error for
  2316. // any of them to be called anywhere except from within constructors.
  2317. void priv_terminate_string()
  2318. { this->priv_construct_null(this->priv_end_addr()); }
  2319. template<class FwdIt, class Count> inline
  2320. void priv_uninitialized_fill_n(FwdIt first, Count count, const CharT val)
  2321. {
  2322. //Save initial position
  2323. FwdIt init = first;
  2324. BOOST_TRY{
  2325. //Construct objects
  2326. for (; count--; ++first){
  2327. this->construct(first, val);
  2328. }
  2329. }
  2330. BOOST_CATCH(...){
  2331. //Call destructors
  2332. for (; init != first; ++init){
  2333. this->destroy(init);
  2334. }
  2335. BOOST_RETHROW
  2336. }
  2337. BOOST_CATCH_END
  2338. }
  2339. template<class InpIt, class FwdIt> inline
  2340. size_type priv_uninitialized_copy(InpIt first, InpIt last, FwdIt dest)
  2341. {
  2342. //Save initial destination position
  2343. FwdIt dest_init = dest;
  2344. size_type constructed = 0;
  2345. BOOST_TRY{
  2346. //Try to build objects
  2347. for (; first != last; ++dest, ++first, ++constructed){
  2348. this->construct(dest, *first);
  2349. }
  2350. }
  2351. BOOST_CATCH(...){
  2352. //Call destructors
  2353. for (; constructed--; ++dest_init){
  2354. this->destroy(dest_init);
  2355. }
  2356. BOOST_RETHROW
  2357. }
  2358. BOOST_CATCH_END
  2359. return (constructed);
  2360. }
  2361. template <class InputIterator, class OutIterator>
  2362. void priv_copy(InputIterator first, InputIterator last, OutIterator result)
  2363. {
  2364. for ( ; first != last; ++first, ++result)
  2365. Traits::assign(*result, *first);
  2366. }
  2367. void priv_copy(const CharT* first, const CharT* last, CharT* result)
  2368. { Traits::copy(result, first, last - first); }
  2369. template <class Integer>
  2370. basic_string& priv_replace_dispatch(const_iterator first, const_iterator last,
  2371. Integer n, Integer x,
  2372. container_detail::true_)
  2373. { return this->replace(first, last, (size_type) n, (CharT) x); }
  2374. template <class InputIter>
  2375. basic_string& priv_replace_dispatch(const_iterator first, const_iterator last,
  2376. InputIter f, InputIter l,
  2377. container_detail::false_)
  2378. {
  2379. typedef typename boost::container::iterator_traits<InputIter>::iterator_category Category;
  2380. return this->priv_replace(first, last, f, l, Category());
  2381. }
  2382. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2383. };
  2384. #ifdef BOOST_CONTAINER_DOXYGEN_INVOKED
  2385. //!Typedef for a basic_string of
  2386. //!narrow characters
  2387. typedef basic_string
  2388. <char
  2389. ,std::char_traits<char>
  2390. ,new_allocator<char> >
  2391. string;
  2392. //!Typedef for a basic_string of
  2393. //!narrow characters
  2394. typedef basic_string
  2395. <wchar_t
  2396. ,std::char_traits<wchar_t>
  2397. ,new_allocator<wchar_t> >
  2398. wstring;
  2399. #endif
  2400. // ------------------------------------------------------------
  2401. // Non-member functions.
  2402. // Operator+
  2403. template <class CharT, class Traits, class Allocator> inline
  2404. basic_string<CharT,Traits,Allocator>
  2405. operator+(const basic_string<CharT,Traits,Allocator>& x
  2406. ,const basic_string<CharT,Traits,Allocator>& y)
  2407. {
  2408. typedef basic_string<CharT,Traits,Allocator> str_t;
  2409. typedef typename str_t::reserve_t reserve_t;
  2410. reserve_t reserve;
  2411. str_t result(reserve, x.size() + y.size(), x.get_stored_allocator());
  2412. result.append(x);
  2413. result.append(y);
  2414. return result;
  2415. }
  2416. template <class CharT, class Traits, class Allocator> inline
  2417. basic_string<CharT, Traits, Allocator> operator+
  2418. ( BOOST_RV_REF_BEG basic_string<CharT, Traits, Allocator> BOOST_RV_REF_END x
  2419. , BOOST_RV_REF_BEG basic_string<CharT, Traits, Allocator> BOOST_RV_REF_END y)
  2420. {
  2421. x += y;
  2422. return boost::move(x);
  2423. }
  2424. template <class CharT, class Traits, class Allocator> inline
  2425. basic_string<CharT, Traits, Allocator> operator+
  2426. ( BOOST_RV_REF_BEG basic_string<CharT, Traits, Allocator> BOOST_RV_REF_END x
  2427. , const basic_string<CharT,Traits,Allocator>& y)
  2428. {
  2429. x += y;
  2430. return boost::move(x);
  2431. }
  2432. template <class CharT, class Traits, class Allocator> inline
  2433. basic_string<CharT, Traits, Allocator> operator+
  2434. (const basic_string<CharT,Traits,Allocator>& x
  2435. ,BOOST_RV_REF_BEG basic_string<CharT, Traits, Allocator> BOOST_RV_REF_END y)
  2436. {
  2437. y.insert(y.begin(), x.begin(), x.end());
  2438. return boost::move(y);
  2439. }
  2440. template <class CharT, class Traits, class Allocator> inline
  2441. basic_string<CharT, Traits, Allocator> operator+
  2442. (const CharT* s, basic_string<CharT, Traits, Allocator> y)
  2443. {
  2444. y.insert(y.begin(), s, s + Traits::length(s));
  2445. return y;
  2446. }
  2447. template <class CharT, class Traits, class Allocator> inline
  2448. basic_string<CharT,Traits,Allocator> operator+
  2449. (basic_string<CharT,Traits,Allocator> x, const CharT* s)
  2450. {
  2451. x += s;
  2452. return x;
  2453. }
  2454. template <class CharT, class Traits, class Allocator> inline
  2455. basic_string<CharT,Traits,Allocator> operator+
  2456. (CharT c, basic_string<CharT,Traits,Allocator> y)
  2457. {
  2458. y.insert(y.begin(), c);
  2459. return y;
  2460. }
  2461. template <class CharT, class Traits, class Allocator> inline
  2462. basic_string<CharT,Traits,Allocator> operator+
  2463. (basic_string<CharT,Traits,Allocator> x, const CharT c)
  2464. {
  2465. x += c;
  2466. return x;
  2467. }
  2468. // Operator== and operator!=
  2469. template <class CharT, class Traits, class Allocator>
  2470. inline bool
  2471. operator==(const basic_string<CharT,Traits,Allocator>& x,
  2472. const basic_string<CharT,Traits,Allocator>& y)
  2473. {
  2474. return x.size() == y.size() &&
  2475. Traits::compare(x.data(), y.data(), x.size()) == 0;
  2476. }
  2477. template <class CharT, class Traits, class Allocator>
  2478. inline bool
  2479. operator==(const CharT* s, const basic_string<CharT,Traits,Allocator>& y)
  2480. {
  2481. typename basic_string<CharT,Traits,Allocator>::size_type n = Traits::length(s);
  2482. return n == y.size() && Traits::compare(s, y.data(), n) == 0;
  2483. }
  2484. template <class CharT, class Traits, class Allocator>
  2485. inline bool
  2486. operator==(const basic_string<CharT,Traits,Allocator>& x, const CharT* s)
  2487. {
  2488. typename basic_string<CharT,Traits,Allocator>::size_type n = Traits::length(s);
  2489. return x.size() == n && Traits::compare(x.data(), s, n) == 0;
  2490. }
  2491. template <class CharT, class Traits, class Allocator>
  2492. inline bool
  2493. operator!=(const basic_string<CharT,Traits,Allocator>& x,
  2494. const basic_string<CharT,Traits,Allocator>& y)
  2495. { return !(x == y); }
  2496. template <class CharT, class Traits, class Allocator>
  2497. inline bool
  2498. operator!=(const CharT* s, const basic_string<CharT,Traits,Allocator>& y)
  2499. { return !(s == y); }
  2500. template <class CharT, class Traits, class Allocator>
  2501. inline bool
  2502. operator!=(const basic_string<CharT,Traits,Allocator>& x, const CharT* s)
  2503. { return !(x == s); }
  2504. // Operator< (and also >, <=, and >=).
  2505. template <class CharT, class Traits, class Allocator>
  2506. inline bool
  2507. operator<(const basic_string<CharT,Traits,Allocator>& x, const basic_string<CharT,Traits,Allocator>& y)
  2508. {
  2509. return x.compare(y) < 0;
  2510. // return basic_string<CharT,Traits,Allocator>
  2511. // ::s_compare(x.begin(), x.end(), y.begin(), y.end()) < 0;
  2512. }
  2513. template <class CharT, class Traits, class Allocator>
  2514. inline bool
  2515. operator<(const CharT* s, const basic_string<CharT,Traits,Allocator>& y)
  2516. {
  2517. return y.compare(s) > 0;
  2518. // basic_string<CharT,Traits,Allocator>::size_type n = Traits::length(s);
  2519. // return basic_string<CharT,Traits,Allocator>
  2520. // ::s_compare(s, s + n, y.begin(), y.end()) < 0;
  2521. }
  2522. template <class CharT, class Traits, class Allocator>
  2523. inline bool
  2524. operator<(const basic_string<CharT,Traits,Allocator>& x,
  2525. const CharT* s)
  2526. {
  2527. return x.compare(s) < 0;
  2528. // basic_string<CharT,Traits,Allocator>::size_type n = Traits::length(s);
  2529. // return basic_string<CharT,Traits,Allocator>
  2530. // ::s_compare(x.begin(), x.end(), s, s + n) < 0;
  2531. }
  2532. template <class CharT, class Traits, class Allocator>
  2533. inline bool
  2534. operator>(const basic_string<CharT,Traits,Allocator>& x,
  2535. const basic_string<CharT,Traits,Allocator>& y) {
  2536. return y < x;
  2537. }
  2538. template <class CharT, class Traits, class Allocator>
  2539. inline bool
  2540. operator>(const CharT* s, const basic_string<CharT,Traits,Allocator>& y) {
  2541. return y < s;
  2542. }
  2543. template <class CharT, class Traits, class Allocator>
  2544. inline bool
  2545. operator>(const basic_string<CharT,Traits,Allocator>& x, const CharT* s)
  2546. {
  2547. return s < x;
  2548. }
  2549. template <class CharT, class Traits, class Allocator>
  2550. inline bool
  2551. operator<=(const basic_string<CharT,Traits,Allocator>& x,
  2552. const basic_string<CharT,Traits,Allocator>& y)
  2553. {
  2554. return !(y < x);
  2555. }
  2556. template <class CharT, class Traits, class Allocator>
  2557. inline bool
  2558. operator<=(const CharT* s, const basic_string<CharT,Traits,Allocator>& y)
  2559. { return !(y < s); }
  2560. template <class CharT, class Traits, class Allocator>
  2561. inline bool
  2562. operator<=(const basic_string<CharT,Traits,Allocator>& x, const CharT* s)
  2563. { return !(s < x); }
  2564. template <class CharT, class Traits, class Allocator>
  2565. inline bool
  2566. operator>=(const basic_string<CharT,Traits,Allocator>& x,
  2567. const basic_string<CharT,Traits,Allocator>& y)
  2568. { return !(x < y); }
  2569. template <class CharT, class Traits, class Allocator>
  2570. inline bool
  2571. operator>=(const CharT* s, const basic_string<CharT,Traits,Allocator>& y)
  2572. { return !(s < y); }
  2573. template <class CharT, class Traits, class Allocator>
  2574. inline bool
  2575. operator>=(const basic_string<CharT,Traits,Allocator>& x, const CharT* s)
  2576. { return !(x < s); }
  2577. // Swap.
  2578. template <class CharT, class Traits, class Allocator>
  2579. inline void swap(basic_string<CharT,Traits,Allocator>& x, basic_string<CharT,Traits,Allocator>& y)
  2580. { x.swap(y); }
  2581. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2582. // I/O.
  2583. namespace container_detail {
  2584. template <class CharT, class Traits>
  2585. inline bool
  2586. string_fill(std::basic_ostream<CharT, Traits>& os,
  2587. std::basic_streambuf<CharT, Traits>* buf,
  2588. std::size_t n)
  2589. {
  2590. CharT f = os.fill();
  2591. std::size_t i;
  2592. bool ok = true;
  2593. for (i = 0; i < n; i++)
  2594. ok = ok && !Traits::eq_int_type(buf->sputc(f), Traits::eof());
  2595. return ok;
  2596. }
  2597. } //namespace container_detail {
  2598. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2599. template <class CharT, class Traits, class Allocator>
  2600. std::basic_ostream<CharT, Traits>&
  2601. operator<<(std::basic_ostream<CharT, Traits>& os, const basic_string<CharT,Traits,Allocator>& s)
  2602. {
  2603. typename std::basic_ostream<CharT, Traits>::sentry sentry(os);
  2604. bool ok = false;
  2605. if (sentry) {
  2606. ok = true;
  2607. typename basic_string<CharT,Traits,Allocator>::size_type n = s.size();
  2608. typename basic_string<CharT,Traits,Allocator>::size_type pad_len = 0;
  2609. const bool left = (os.flags() & std::ios::left) != 0;
  2610. const std::size_t w = os.width(0);
  2611. std::basic_streambuf<CharT, Traits>* buf = os.rdbuf();
  2612. if (w != 0 && n < w)
  2613. pad_len = w - n;
  2614. if (!left)
  2615. ok = container_detail::string_fill(os, buf, pad_len);
  2616. ok = ok &&
  2617. buf->sputn(s.data(), std::streamsize(n)) == std::streamsize(n);
  2618. if (left)
  2619. ok = ok && container_detail::string_fill(os, buf, pad_len);
  2620. }
  2621. if (!ok)
  2622. os.setstate(std::ios_base::failbit);
  2623. return os;
  2624. }
  2625. template <class CharT, class Traits, class Allocator>
  2626. std::basic_istream<CharT, Traits>&
  2627. operator>>(std::basic_istream<CharT, Traits>& is, basic_string<CharT,Traits,Allocator>& s)
  2628. {
  2629. typename std::basic_istream<CharT, Traits>::sentry sentry(is);
  2630. if (sentry) {
  2631. std::basic_streambuf<CharT, Traits>* buf = is.rdbuf();
  2632. const std::ctype<CharT>& ctype = std::use_facet<std::ctype<CharT> >(is.getloc());
  2633. s.clear();
  2634. std::size_t n = is.width(0);
  2635. if (n == 0)
  2636. n = static_cast<std::size_t>(-1);
  2637. else
  2638. s.reserve(n);
  2639. while (n-- > 0) {
  2640. typename Traits::int_type c1 = buf->sbumpc();
  2641. if (Traits::eq_int_type(c1, Traits::eof())) {
  2642. is.setstate(std::ios_base::eofbit);
  2643. break;
  2644. }
  2645. else {
  2646. CharT c = Traits::to_char_type(c1);
  2647. if (ctype.is(std::ctype<CharT>::space, c)) {
  2648. if (Traits::eq_int_type(buf->sputbackc(c), Traits::eof()))
  2649. is.setstate(std::ios_base::failbit);
  2650. break;
  2651. }
  2652. else
  2653. s.push_back(c);
  2654. }
  2655. }
  2656. // If we have read no characters, then set failbit.
  2657. if (s.size() == 0)
  2658. is.setstate(std::ios_base::failbit);
  2659. }
  2660. else
  2661. is.setstate(std::ios_base::failbit);
  2662. return is;
  2663. }
  2664. template <class CharT, class Traits, class Allocator>
  2665. std::basic_istream<CharT, Traits>&
  2666. getline(std::istream& is, basic_string<CharT,Traits,Allocator>& s,CharT delim)
  2667. {
  2668. typename basic_string<CharT,Traits,Allocator>::size_type nread = 0;
  2669. typename std::basic_istream<CharT, Traits>::sentry sentry(is, true);
  2670. if (sentry) {
  2671. std::basic_streambuf<CharT, Traits>* buf = is.rdbuf();
  2672. s.clear();
  2673. while (nread < s.max_size()) {
  2674. int c1 = buf->sbumpc();
  2675. if (Traits::eq_int_type(c1, Traits::eof())) {
  2676. is.setstate(std::ios_base::eofbit);
  2677. break;
  2678. }
  2679. else {
  2680. ++nread;
  2681. CharT c = Traits::to_char_type(c1);
  2682. if (!Traits::eq(c, delim))
  2683. s.push_back(c);
  2684. else
  2685. break; // Character is extracted but not appended.
  2686. }
  2687. }
  2688. }
  2689. if (nread == 0 || nread >= s.max_size())
  2690. is.setstate(std::ios_base::failbit);
  2691. return is;
  2692. }
  2693. template <class CharT, class Traits, class Allocator>
  2694. inline std::basic_istream<CharT, Traits>&
  2695. getline(std::basic_istream<CharT, Traits>& is, basic_string<CharT,Traits,Allocator>& s)
  2696. {
  2697. return getline(is, s, '\n');
  2698. }
  2699. template <class Ch, class Allocator>
  2700. inline std::size_t hash_value(basic_string<Ch, std::char_traits<Ch>, Allocator> const& v)
  2701. {
  2702. return hash_range(v.begin(), v.end());
  2703. }
  2704. }}
  2705. #ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2706. namespace boost {
  2707. //!has_trivial_destructor_after_move<> == true_type
  2708. //!specialization for optimizations
  2709. template <class C, class T, class Allocator>
  2710. struct has_trivial_destructor_after_move<boost::container::basic_string<C, T, Allocator> >
  2711. {
  2712. typedef typename ::boost::container::allocator_traits<Allocator>::pointer pointer;
  2713. static const bool value = ::boost::has_trivial_destructor_after_move<Allocator>::value &&
  2714. ::boost::has_trivial_destructor_after_move<pointer>::value;
  2715. };
  2716. }
  2717. #endif //#ifndef BOOST_CONTAINER_DOXYGEN_INVOKED
  2718. #include <boost/container/detail/config_end.hpp>
  2719. #endif // BOOST_CONTAINER_STRING_HPP