| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913 | //// Copyright (c) 2019 Vinnie Falco (vinnie.falco@gmail.com)// Copyright (c) 2020 Krystian Stasiowski (sdkrystian@gmail.com)//// Distributed under the Boost Software License, Version 1.0. (See accompanying// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)//// Official repository: https://github.com/boostorg/json//#ifndef BOOST_JSON_STRING_HPP#define BOOST_JSON_STRING_HPP#include <boost/json/detail/config.hpp>#include <boost/json/pilfer.hpp>#include <boost/json/storage_ptr.hpp>#include <boost/json/string_view.hpp>#include <boost/json/detail/digest.hpp>#include <boost/json/detail/except.hpp>#include <boost/json/detail/string_impl.hpp>#include <boost/json/detail/value.hpp>#include <algorithm>#include <cstring>#include <initializer_list>#include <iosfwd>#include <iterator>#include <limits>#include <new>#include <type_traits>#include <utility>namespace boost {namespace json {class value;/** The native type of string values.    Instances of string store and manipulate sequences    of `char` using the UTF-8 encoding. The elements of    a string are stored contiguously. A pointer to any    character in a string may be passed to functions    that expect a pointer to the first element of a    null-terminated `char` array. The type uses small    buffer optimisation to avoid allocations for small    strings.    String iterators are regular `char` pointers.    @note `string` member functions do not validate    any UTF-8 byte sequences passed to them.    @par Thread Safety    Non-const member functions may not be called    concurrently with any other member functions.    @par Satisfies        <a href="https://en.cppreference.com/w/cpp/named_req/ContiguousContainer"><em>ContiguousContainer</em></a>,        <a href="https://en.cppreference.com/w/cpp/named_req/ReversibleContainer"><em>ReversibleContainer</em></a>, and        <a href="https://en.cppreference.com/w/cpp/named_req/SequenceContainer"><em>SequenceContainer</em></a>.*/class string{    friend class value;#ifndef BOOST_JSON_DOCS    // VFALCO doc toolchain shouldn't show this but does    friend struct detail::access;#endif    using string_impl = detail::string_impl;    inline    string(        detail::key_t const&,        string_view s,        storage_ptr sp);    inline    string(        detail::key_t const&,        string_view s1,        string_view s2,        storage_ptr sp);public:    /// Associated [Allocator](https://en.cppreference.com/w/cpp/named_req/Allocator)    using allocator_type = container::pmr::polymorphic_allocator<value>;    /// The type of a character    using value_type        = char;    /// The type used to represent unsigned integers    using size_type         = std::size_t;    /// The type used to represent signed integers    using difference_type   = std::ptrdiff_t;    /// A pointer to an element    using pointer           = char*;    /// A const pointer to an element    using const_pointer     = char const*;    /// A reference to an element    using reference         = char&;    /// A const reference to an element    using const_reference   = const char&;    /// A random access iterator to an element    using iterator          = char*;    /// A random access const iterator to an element    using const_iterator    = char const*;    /// A reverse random access iterator to an element    using reverse_iterator =        std::reverse_iterator<iterator>;    /// A reverse random access const iterator to an element    using const_reverse_iterator =        std::reverse_iterator<const_iterator>;    /// A special index    static constexpr std::size_t npos =        string_view::npos;private:    template<class T>    using is_inputit = typename std::enable_if<        std::is_convertible<typename            std::iterator_traits<T>::reference,            char>::value>::type;    storage_ptr sp_; // must come first    string_impl impl_;public:    /** Destructor.        Any dynamically allocated internal storage        is freed.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    ~string() noexcept    {        impl_.destroy(sp_);    }    //------------------------------------------------------    //    // Construction    //    //------------------------------------------------------    /** Default constructor.        The string will have a zero size and a non-zero,        unspecified capacity, using the [default memory resource].        @par Complexity        Constant.        [default memory resource]: json/allocators/storage_ptr.html#json.allocators.storage_ptr.default_memory_resource    */    string() = default;    /** Pilfer constructor.        The string is constructed by acquiring ownership        of the contents of `other` using pilfer semantics.        This is more efficient than move construction, when        it is known that the moved-from object will be        immediately destroyed afterwards.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.        @param other The value to pilfer. After pilfer        construction, `other` is not in a usable state        and may only be destroyed.        @see @ref pilfer,            <a href="http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2016/p0308r0.html">                Valueless Variants Considered Harmful</a>    */    string(pilfered<string> other) noexcept        : sp_(std::move(other.get().sp_))        , impl_(other.get().impl_)    {        ::new(&other.get().impl_) string_impl();    }    /** Constructor.        The string will have zero size and a non-zero,        unspecified capacity, obtained from the specified        memory resource.        @par Complexity        Constant.        @param sp A pointer to the `boost::container::pmr::memory_resource` to        use. The container will acquire shared ownership of the memory        resource.    */    explicit    string(storage_ptr sp)        : sp_(std::move(sp))    {    }    /** Constructor.        Construct the contents with `count` copies of        character `ch`.        @par Complexity        Linear in `count`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param count The size of the resulting string.        @param ch The value to initialize characters        of the string with.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.        @throw `boost::system::system_error` `count > max_size()`.    */    BOOST_JSON_DECL    explicit    string(        std::size_t count,        char ch,        storage_ptr sp = {});    /** Constructor.        Construct the contents with those of the null        terminated string pointed to by `s`. The length        of the string is determined by the first null        character.        @par Complexity        Linear in `strlen(s)`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param s A pointer to a character string used to        copy from.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.        @throw `boost::system::system_error` `strlen(s) > max_size()`.    */    BOOST_JSON_DECL    string(        char const* s,        storage_ptr sp = {});    /** Constructor.        Construct the contents with copies of the        characters in the range `{s, s+count)`.        This range can contain null characters.        @par Complexity        Linear in `count`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param count The number of characters to copy.        @param s A pointer to a character string used to        copy from.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.        @throw `boost::system::system_error` `count > max_size()`.    */    BOOST_JSON_DECL    explicit    string(        char const* s,        std::size_t count,        storage_ptr sp = {});    /** Constructor.        Construct the contents with copies of characters        in the range `{first, last)`.        @par Complexity        Linear in `std::distance(first, last)`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @tparam InputIt The type of the iterators.        @par Constraints        `InputIt` satisfies __InputIterator__.        @param first An input iterator pointing to the        first character to insert, or pointing to the        end of the range.        @param last An input iterator pointing to the end        of the range.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.        @throw `boost::system::system_error` `std::distance(first, last) > max_size()`.    */    template<class InputIt    #ifndef BOOST_JSON_DOCS        ,class = is_inputit<InputIt>    #endif    >    explicit    string(        InputIt first,        InputIt last,        storage_ptr sp = {});    /** Copy constructor.        Construct the contents with a copy of `other`.        @par Complexity        Linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param other The string to use as a source        to copy from.    */    BOOST_JSON_DECL    string(string const& other);    /** Constructor.        Construct the contents with a copy of `other`.        @par Complexity        Linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param other The string to use as a source        to copy from.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.    */    BOOST_JSON_DECL    explicit    string(        string const& other,        storage_ptr sp);    /** Move constructor.        Constructs the string with the contents of `other` using move        semantics. Ownership of the underlying memory is transferred. The        container acquires shared ownership of the        `boost::container::pmr::memory_resource` used by `other`. After        construction, the moved-from string behaves as if newly constructed        with its current memory resource.        @par Complexity        Constant.        @param other The string to move    */    string(string&& other) noexcept        : sp_(other.sp_)        , impl_(other.impl_)    {        ::new(&other.impl_) string_impl();    }    /** Constructor.        Construct the contents with those of `other`        using move semantics.        @li If `*other.storage() == *sp`, ownership of the underlying memory is        transferred in constant time, with no possibility of exceptions. After        construction, the moved-from string behaves as if newly constructed        with its current `boost::container::pmr::memory_resource`. Otherwise,        @li If `*other.storage() != *sp`,        a copy of the characters in `other` is made. In        this case, the moved-from string is not changed.        @par Complexity        Constant or linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param other The string to assign from.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.    */    BOOST_JSON_DECL    explicit    string(        string&& other,        storage_ptr sp);    /** Constructor.        Construct the contents with those of a        string view. This view can contain        null characters.        @par Complexity        Linear in `s.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param s The string view to copy from.        @param sp An optional pointer to the        `boost::container::pmr::memory_resource` to use. The container will        acquire shared ownership of the memory resource. The default argument        for this parameter is `{}`.        @throw `boost::system::system_error` `std::distance(first, last) > max_size()`.    */    BOOST_JSON_DECL    string(        string_view s,        storage_ptr sp = {});    //------------------------------------------------------    //    // Assignment    //    //------------------------------------------------------    /** Copy assignment.        Replace the contents with a copy of `other`.        @par Complexity        Linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param other The string to use as a source        to copy from.    */    BOOST_JSON_DECL    string&    operator=(string const& other);    /** Move assignment.        Replace the contents with those of `other`        using move semantics.        @li If `&other == this`, do nothing. Otherwise,        @li If `*other.storage() == *this->storage()`, ownership of the        underlying memory is transferred in constant time, with no possibility        of exceptions. After construction, the moved-from string behaves as if        newly constructed with its current        `boost::container::pmr::memory_resource`. Otherwise,        @li a copy of the characters in `other` is made. In        this case, the moved-from container is not changed.        @par Complexity        Constant or linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param other The string to use as a source        to move from.    */    BOOST_JSON_DECL    string&    operator=(string&& other);    /** Assign a value to the string.        Replaces the contents with those of the null        terminated string pointed to by `s`. The length        of the string is determined by the first null        character.        @par Complexity        Linear in `std::strlen(s)`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param s The null-terminated character string.        @throw `boost::system::system_error` `std::strlen(s) > max_size()`.    */    BOOST_JSON_DECL    string&    operator=(char const* s);    /** Assign a value to the string.        Replaces the contents with those of a        string view. This view can contain        null characters.        @par Complexity        Linear in `s.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param s The string view to copy from.        @throw `boost::system::system_error` `s.size() > max_size()`.    */    BOOST_JSON_DECL    string&    operator=(string_view s);    //------------------------------------------------------    /** Assign characters to a string.        Replace the contents with `count` copies of        character `ch`.        @par Complexity        Linear in `count`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param count The size of the resulting string.        @param ch The value to initialize characters        of the string with.        @throw `boost::system::system_error` `count > max_size()`.    */    BOOST_JSON_DECL    string&    assign(        std::size_t count,        char ch);    /** Assign characters to a string.        Replace the contents with a copy of `other`.        @par Complexity        Linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param other The string to use as a source        to copy from.    */    BOOST_JSON_DECL    string&    assign(        string const& other);    /** Assign characters to a string.        Replace the contents with those of `other`        using move semantics.        @li If `&other == this`, do nothing. Otherwise,        @li If `*other.storage() == *this->storage()`, ownership of the        underlying memory is transferred in constant time, with no possibility        of exceptions. After construction, the moved-from string behaves as if        newly constructed with its current        `boost::container::pmr::memory_resource`, otherwise        @li If `*other.storage() != *this->storage()`,        a copy of the characters in `other` is made.        In this case, the moved-from container        is not changed.        @par Complexity        Constant or linear in `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param other The string to assign from.    */    BOOST_JSON_DECL    string&    assign(string&& other);    /** Assign characters to a string.        Replaces the contents with copies of the        characters in the range `{s, s+count)`. This        range can contain null characters.        @par Complexity        Linear in `count`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param count The number of characters to copy.        @param s A pointer to a character string used to        copy from.        @throw `boost::system::system_error` `count > max_size()`.    */    BOOST_JSON_DECL    string&    assign(        char const* s,        std::size_t count);    /** Assign characters to a string.        Replaces the contents with those of the null        terminated string pointed to by `s`. The length        of the string is determined by the first null        character.        @par Complexity        Linear in `strlen(s)`.        @par Exception Safety        Strong guarantee.        @note        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param s A pointer to a character string used to        copy from.        @throw `boost::system::system_error` `strlen(s) > max_size()`.    */    BOOST_JSON_DECL    string&    assign(        char const* s);    /** Assign characters to a string.        Replaces the contents with copies of characters        in the range `{first, last)`.        @par Complexity        Linear in `std::distance(first, last)`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @tparam InputIt The type of the iterators.        @par Constraints        `InputIt` satisfies __InputIterator__.        @return `*this`        @param first An input iterator pointing to the        first character to insert, or pointing to the        end of the range.        @param last An input iterator pointing to the end        of the range.        @throw `boost::system::system_error` `std::distance(first, last) > max_size()`.    */    template<class InputIt    #ifndef BOOST_JSON_DOCS        ,class = is_inputit<InputIt>    #endif    >    string&    assign(        InputIt first,        InputIt last);    /** Assign characters to a string.        Replaces the contents with those of a        string view. This view can contain        null characters.        @par Complexity        Linear in `s.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @return `*this`        @param s The string view to copy from.        @throw `boost::system::system_error` `s.size() > max_size()`.    */    string&    assign(string_view s)    {        return assign(s.data(), s.size());    }    //------------------------------------------------------    /** Return the associated memory resource.        This returns the `boost::container::pmr::memory_resource` used by        the container.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    storage_ptr const&    storage() const noexcept    {        return sp_;    }    /** Return the associated allocator.        This function returns an instance of @ref allocator_type constructed        from the associated `boost::container::pmr::memory_resource`.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    allocator_type    get_allocator() const noexcept    {        return sp_.get();    }    //------------------------------------------------------    //    // Element Access    //    //------------------------------------------------------    /** Return a character with bounds checking.        Returns `boost::system::result` containing a reference to the character        specified at location `pos`, if `pos` is within the range of the        string. Otherwise the result contains an `error_code`.        @par Exception Safety        Strong guarantee.        @param pos A zero-based index to access.        @par Complexity        Constant.    */    /** @{ */    BOOST_JSON_DECL    system::result<char&>    try_at(std::size_t pos) noexcept;    BOOST_JSON_DECL    system::result<char const&>    try_at(std::size_t pos) const noexcept;    /** @} */    /** Return a character with bounds checking.        Returns a reference to the character specified at        location `pos`.        @par Complexity        Constant.        @par Exception Safety        Strong guarantee.        @param pos A zero-based index to access.        @param loc `source_location` to use in thrown exception; the source            location of the call site by default.        @throw `boost::system::system_error` `pos >= size()`.    */    /** @{ */    inline    char&    at(        std::size_t pos,        source_location const& loc = BOOST_CURRENT_LOCATION);    BOOST_JSON_DECL    char const&    at(        std::size_t pos,        source_location const& loc = BOOST_CURRENT_LOCATION) const;    /** @} */    /** Return a character without bounds checking.        Returns a reference to the character specified at        location `pos`.        @par Complexity        Constant.        @par Precondition        @code        pos >= size        @endcode        @param pos A zero-based index to access.    */    char&    operator[](std::size_t pos)    {        return impl_.data()[pos];    }   /**  Return a character without bounds checking.        Returns a reference to the character specified at        location `pos`.        @par Complexity        Constant.        @par Precondition        @code        pos >= size        @endcode        @param pos A zero-based index to access.    */    const char&    operator[](std::size_t pos) const    {        return impl_.data()[pos];    }    /** Return the first character.        Returns a reference to the first character.        @par Complexity        Constant.        @par Precondition        @code        not empty()        @endcode    */    char&    front()    {        return impl_.data()[0];    }    /** Return the first character.        Returns a reference to the first character.        @par Complexity        Constant.        @par Precondition        @code        not empty()        @endcode    */    char const&    front() const    {        return impl_.data()[0];    }    /** Return the last character.        Returns a reference to the last character.        @par Complexity        Constant.        @par Precondition        @code        not empty()        @endcode    */    char&    back()    {        return impl_.data()[impl_.size() - 1];    }    /** Return the last character.        Returns a reference to the last character.        @par Complexity        Constant.        @par Precondition        @code        not empty()        @endcode    */    char const&    back() const    {        return impl_.data()[impl_.size() - 1];    }    /** Return the underlying character array directly.        Returns a pointer to the underlying array        serving as storage. The value returned is such that        the range `{data(), data()+size())` is always a        valid range, even if the container is empty.        @par Complexity        Constant.        @note The value returned from        this function is never equal to `nullptr`.    */    char*    data() noexcept    {        return impl_.data();    }    /** Return the underlying character array directly.        Returns a pointer to the underlying array        serving as storage.        @note The value returned is such that        the range `{data(), data() + size())` is always a        valid range, even if the container is empty.        The value returned from        this function is never equal to `nullptr`.        @par Complexity        Constant.    */    char const*    data() const noexcept    {        return impl_.data();    }    /** Return the underlying character array directly.        Returns a pointer to the underlying array        serving as storage. The value returned is such that        the range `{c_str(), c_str() + size()}` is always a        valid range, even if the container is empty.        @par Complexity        Constant.        @note The value returned from        this function is never equal to `nullptr`.    */    char const*    c_str() const noexcept    {        return impl_.data();    }    /** Convert to a @ref string_view referring to the string.        Returns a string view to the        underlying character string. The size of the view        does not include the null terminator.        @par Complexity        Constant.    */    operator string_view() const noexcept    {        return {data(), size()};    }#if ! defined(BOOST_NO_CXX17_HDR_STRING_VIEW)    /** Convert to a `std::string_view` referring to the string.        Returns a string view to the underlying character string. The size of        the view does not include the null terminator.        This overload is not defined when `BOOST_NO_CXX17_HDR_STRING_VIEW`        is defined.        @par Complexity        Constant.    */    operator std::string_view() const noexcept    {        return {data(), size()};    }#endif    //------------------------------------------------------    //    // Iterators    //    //------------------------------------------------------    /** Return an iterator to the beginning.        If the container is empty, @ref end() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    iterator    begin() noexcept    {        return impl_.data();    }    /** Return an iterator to the beginning.        If the container is empty, @ref end() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_iterator    begin() const noexcept    {        return impl_.data();    }    /** Return an iterator to the beginning.        If the container is empty, @ref cend() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_iterator    cbegin() const noexcept    {        return impl_.data();    }    /** Return an iterator to the end.        Returns an iterator to the character        following the last character of the string.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    iterator    end() noexcept    {        return impl_.end();    }    /** Return an iterator to the end.        Returns an iterator to the character following        the last character of the string.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_iterator    end() const noexcept    {        return impl_.end();    }    /** Return an iterator to the end.        Returns an iterator to the character following        the last character of the string.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_iterator    cend() const noexcept    {        return impl_.end();    }    /** Return a reverse iterator to the first character of the reversed container.        Returns the pointed-to character that        corresponds to the last character of the        non-reversed container.        If the container is empty, @ref rend() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    reverse_iterator    rbegin() noexcept    {        return reverse_iterator(impl_.end());    }    /** Return a reverse iterator to the first character of the reversed container.        Returns the pointed-to character that        corresponds to the last character of the        non-reversed container.        If the container is empty, @ref rend() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_reverse_iterator    rbegin() const noexcept    {        return const_reverse_iterator(impl_.end());    }    /** Return a reverse iterator to the first character of the reversed container.        Returns the pointed-to character that        corresponds to the last character of the        non-reversed container.        If the container is empty, @ref crend() is returned.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_reverse_iterator    crbegin() const noexcept    {        return const_reverse_iterator(impl_.end());    }    /** Return a reverse iterator to the character following the last character of the reversed container.        Returns the pointed-to character that corresponds        to the character preceding the first character of        the non-reversed container.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    reverse_iterator    rend() noexcept    {        return reverse_iterator(begin());    }    /** Return a reverse iterator to the character following the last character of the reversed container.        Returns the pointed-to character that corresponds        to the character preceding the first character of        the non-reversed container.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_reverse_iterator    rend() const noexcept    {        return const_reverse_iterator(begin());    }    /** Return a reverse iterator to the character following the last character of the reversed container.        Returns the pointed-to character that corresponds        to the character preceding the first character of        the non-reversed container.        This character acts as a placeholder, attempting        to access it results in undefined behavior.        @par Complexity        Constant.        @par Exception Safety        No-throw guarantee.    */    const_reverse_iterator    crend() const noexcept    {        return const_reverse_iterator(begin());    }    //------------------------------------------------------    //    // Capacity    //    //------------------------------------------------------    /** Check if the string has no characters.        Returns `true` if there are no characters in        the string, i.e. @ref size() returns 0.        @par Complexity        Constant.    */    bool    empty() const noexcept    {        return impl_.size() == 0;    }    /** Return the number of characters in the string.        The value returned does not include the        null terminator, which is always present.        @par Complexity        Constant.    */    std::size_t    size() const noexcept    {        return impl_.size();    }    /** Return the maximum number of characters any string can hold.        The maximum is an implementation-defined number.        This value is a theoretical limit; at runtime,        the actual maximum size may be less due to        resource limits.        @par Complexity        Constant.    */    static    constexpr    std::size_t    max_size() noexcept    {        return string_impl::max_size();    }    /** Return the number of characters that can be held without a reallocation.        This number represents the largest number of        characters the currently allocated storage can contain.        This number may be larger than the value returned        by @ref size().        @par Complexity        Constant.    */    std::size_t    capacity() const noexcept    {        return impl_.capacity();    }    /** Increase the capacity to at least a certain amount.        This increases the capacity of the array to a value        that is greater than or equal to `new_capacity`. If        `new_capacity > capacity()`, new memory is        allocated. Otherwise, the call has no effect.        The number of elements and therefore the        @ref size() of the container is not changed.        @par Complexity        At most, linear in @ref size().        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @note        If new memory is allocated, all iterators including        any past-the-end iterators, and all references to        the elements are invalidated. Otherwise, no        iterators or references are invalidated.        @param new_capacity The new capacity of the array.        @throw `boost::system::system_error` `new_capacity > max_size()`.    */    void    reserve(std::size_t new_capacity)    {        if(new_capacity <= capacity())            return;        reserve_impl(new_capacity);    }    /** Request the removal of unused capacity.        This performs a non-binding request to reduce        @ref capacity() to @ref size(). The request may        or may not be fulfilled.        @par Complexity        At most, linear in @ref size().        @note If reallocation occurs, all iterators        including  any past-the-end iterators, and all        references to characters are invalidated.        Otherwise, no iterators or references are        invalidated.    */    BOOST_JSON_DECL    void    shrink_to_fit();    //------------------------------------------------------    //    // Operations    //    //------------------------------------------------------    /** Clear the contents.        Erases all characters from the string. After this        call, @ref size() returns zero but @ref capacity()        is unchanged.        @par Complexity        Linear in @ref size().        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.    */    BOOST_JSON_DECL    void    clear() noexcept;    //------------------------------------------------------    /** Insert a string.        Inserts the `string_view` `sv` at the position `pos`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to insert at.        @param sv The `string_view` to insert.        @throw `boost::system::system_error` `size() + s.size() > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    BOOST_JSON_DECL    string&    insert(        std::size_t pos,        string_view sv);    /** Insert a character.        Inserts `count` copies of `ch` at the position `pos`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to insert at.        @param count The number of characters to insert.        @param ch The character to insert.        @throw `boost::system::system_error` `size() + count > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    BOOST_JSON_DECL    string&    insert(        std::size_t pos,        std::size_t count,        char ch);    /** Insert a character.        Inserts the character `ch` before the character        at index `pos`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to insert at.        @param ch The character to insert.        @throw `boost::system::system_error` `size() + 1 > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    string&    insert(        size_type pos,        char ch)    {        return insert(pos, 1, ch);    }    /** Insert a range of characters.        Inserts characters from the range `{first, last)`        before the character at index `pos`.        @par Precondition        `{first, last)` is a valid range.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @tparam InputIt The type of the iterators.        @par Constraints        `InputIt` satisfies __InputIterator__.        @return `*this`        @param pos The index to insert at.        @param first The beginning of the character range.        @param last The end of the character range.        @throw `boost::system::system_error` `size() + insert_count > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    template<class InputIt    #ifndef BOOST_JSON_DOCS        ,class = is_inputit<InputIt>    #endif    >    string&    insert(        size_type pos,        InputIt first,        InputIt last);    //------------------------------------------------------    /** Erase characters from the string.        Erases `num` characters from the string, starting        at `pos`.  `num` is determined as the smaller of        `count` and `size() - pos`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to erase at.        The default argument for this parameter is `0`.        @param count The number of characters to erase.        The default argument for this parameter        is @ref npos.        @throw `boost::system::system_error` `pos > size()`.    */    BOOST_JSON_DECL    string&    erase(        std::size_t pos = 0,        std::size_t count = npos);    /** Erase a character from the string.        Erases the character at `pos`.        @par Precondition        @code        pos >= data() && pos <= data() + size()        @endcode        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return An iterator referring to character        immediately following the erased character, or        @ref end() if one does not exist.        @param pos An iterator referring to the        character to erase.    */    BOOST_JSON_DECL    iterator    erase(const_iterator pos);    /** Erase a range from the string.        Erases the characters in the range `{first, last)`.        @par Precondition        `{first, last}` shall be valid within        @code        {data(), data() + size()}        @endcode        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return An iterator referring to the character        `last` previously referred to, or @ref end()        if one does not exist.        @param first An iterator representing the first        character to erase.        @param last An iterator one past the last        character to erase.    */    BOOST_JSON_DECL    iterator    erase(        const_iterator first,        const_iterator last);    //------------------------------------------------------    /** Append a character.        Appends a character to the end of the string.        @par Exception Safety        Strong guarantee.        @param ch The character to append.        @throw `boost::system::system_error` `size() + 1 > max_size()`.    */    BOOST_JSON_DECL    void    push_back(char ch);    /** Remove the last character.        Removes a character from the end of the string.        @par Precondition        @code        not empty()        @endcode    */    BOOST_JSON_DECL    void    pop_back();    //------------------------------------------------------    /** Append characters to the string.        Appends `count` copies of `ch` to the end of        the string.        @par Exception Safety        Strong guarantee.        @return `*this`        @param count The number of characters to append.        @param ch The character to append.        @throw `boost::system::system_error` `size() + count > max_size()`.    */    BOOST_JSON_DECL    string&    append(        std::size_t count,        char ch);    /** Append a string to the string.        Appends `sv` the end of the string.        @par Exception Safety        Strong guarantee.        @return `*this`        @param sv The `string_view` to append.        @throw `boost::system::system_error` `size() + s.size() > max_size()`.    */    BOOST_JSON_DECL    string&    append(string_view sv);    /** Append a range of characters.        Appends characters from the range `{first, last)`        to the end of the string.        @par Precondition        `{first, last)` shall be a valid range        @par Exception Safety        Strong guarantee.        @tparam InputIt The type of the iterators.        @par Constraints        `InputIt` satisfies __InputIterator__.        @return `*this`        @param first An iterator representing the        first character to append.        @param last An iterator one past the        last character to append.        @throw `boost::system::system_error` `size() + insert_count > max_size()`.    */    template<class InputIt    #ifndef BOOST_JSON_DOCS        ,class = is_inputit<InputIt>    #endif    >    string&    append(InputIt first, InputIt last);    //------------------------------------------------------    /** Append characters from a string.        Appends `{sv.begin(), sv.end())` to the end of        the string.        @par Exception Safety        Strong guarantee.        @return `*this`        @param sv The `string_view` to append.        @throw `boost::system::system_error` `size() + sv.size() > max_size()`.    */    string&    operator+=(string_view sv)    {        return append(sv);    }    /** Append a character.        Appends a character to the end of the string.        @par Exception Safety        Strong guarantee.        @param ch The character to append.        @throw `boost::system::system_error` `size() + 1 > max_size()`.    */    string&    operator+=(char ch)    {        push_back(ch);        return *this;    }    //------------------------------------------------------    /** Compare a string with the string.        Let `comp` be        `std::char_traits<char>::compare(data(), sv.data(), std::min(size(), sv.size())`.        If `comp != 0`, then the result is `comp`. Otherwise,        the result is `0` if `size() == sv.size()`,        `-1` if `size() < sv.size()`, and `1` otherwise.        @par Complexity        Linear.        @return The result of lexicographically comparing        the characters of `sv` and the string.        @param sv The `string_view` to compare.    */    int    compare(string_view sv) const noexcept    {        return subview().compare(sv);    }    //------------------------------------------------------    /** Return whether the string begins with a string.        Returns `true` if the string begins with `s`,        and `false` otherwise.        @par Complexity        Linear.        @param s The `string_view` to check for.    */    bool    starts_with(string_view s) const noexcept    {        return subview(0, s.size()) == s;    }    /** Return whether the string begins with a character.        Returns `true` if the string begins with `ch`,        and `false` otherwise.        @par Complexity        Constant.        @param ch The character to check for.    */    bool    starts_with(char ch) const noexcept    {        return ! empty() && front() == ch;    }    /** Return whether the string end with a string.        Returns `true` if the string end with `s`,        and `false` otherwise.        @par Complexity        Linear.        @param s The string to check for.    */    bool    ends_with(string_view s) const noexcept    {        return size() >= s.size() &&            subview(size() - s.size()) == s;    }    /** Return whether the string ends with a character.        Returns `true` if the string ends with `ch`,        and `false` otherwise.        @par Complexity        Constant.        @param ch The character to check for.    */    bool    ends_with(char ch) const noexcept    {        return ! empty() && back() == ch;    }    //------------------------------------------------------    /** Replace a substring with a string.        Replaces `rcount` characters starting at index        `pos` with those of `sv`, where `rcount` is        `std::min(count, size() - pos)`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to replace at.        @param count The number of characters to replace.        @param sv The `string_view` to replace with.        @throw `boost::system::system_error` `size() + (sv.size() - rcount) > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    BOOST_JSON_DECL    string&    replace(        std::size_t pos,        std::size_t count,        string_view sv);    /** Replace a range with a string.        Replaces the characters in the range        `{first, last)` with those of `sv`.        @par Precondition        `{first, last)` is a valid range.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param first An iterator referring to the first        character to replace.        @param last An iterator one past the end of        the last character to replace.        @param sv The `string_view` to replace with.        @throw `boost::system::system_error` `size() + (sv.size() - std::distance(first, last)) > max_size()`.    */    string&    replace(        const_iterator first,        const_iterator last,        string_view sv)    {        return replace(first - begin(), last - first, sv);    }    /** Replace a range with a range.        Replaces the characters in the range        `{first, last)` with those of `{first2, last2)`.        @par Precondition        `{first, last)` is a valid range.        `{first2, last2)` is a valid range.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @tparam InputIt The type of the iterators.        @par Constraints        `InputIt` satisfies __InputIterator__.        @return `*this`        @param first An iterator referring to the first        character to replace.        @param last An iterator one past the end of        the last character to replace.        @param first2 An iterator referring to the first        character to replace with.        @param last2 An iterator one past the end of        the last character to replace with.        @throw `boost::system::system_error` `size() + (inserted - std::distance(first, last)) > max_size()`.    */    template<class InputIt    #ifndef BOOST_JSON_DOCS        ,class = is_inputit<InputIt>    #endif    >    string&    replace(        const_iterator first,        const_iterator last,        InputIt first2,        InputIt last2);    /** Replace a substring with copies of a character.        Replaces `rcount` characters starting at index        `pos`with `count2` copies of `ch`, where        `rcount` is `std::min(count, size() - pos)`.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param pos The index to replace at.        @param count The number of characters to replace.        @param count2 The number of characters to        replace with.        @param ch The character to replace with.        @throw `boost::system::system_error` `size() + (count2 - rcount) > max_size()`.        @throw `boost::system::system_error` `pos > size()`.    */    BOOST_JSON_DECL    string&    replace(        std::size_t pos,        std::size_t count,        std::size_t count2,        char ch);    /** Replace a range with copies of a character.        Replaces the characters in the range        `{first, last)` with `count` copies of `ch`.        @par Precondition        `{first, last)` is a valid range.        @par Exception Safety        Strong guarantee.        @note All references, pointers, or iterators        referring to contained elements are invalidated.        Any past-the-end iterators are also invalidated.        @return `*this`        @param first An iterator referring to the first        character to replace.        @param last An iterator one past the end of        the last character to replace.        @param count The number of characters to        replace with.        @param ch The character to replace with.        @throw `boost::system::system_error` `size() + (count - std::distance(first, last)) > max_size()`.    */    string&    replace(        const_iterator first,        const_iterator last,        std::size_t count,        char ch)    {        return replace(first - begin(), last - first, count, ch);    }    //------------------------------------------------------    /** Return a view.        Returns a view of a substring.        @par Exception Safety        Strong guarantee.        @return `this->subview().substr(pos, count)`        @param pos The index to being the substring at.        The default argument for this parameter is `0`.        @param count The length of the substring.        The default argument for this parameter        is @ref npos.        @throw `boost::system::system_error` `pos > size()`.    */    string_view    subview(        std::size_t pos        ,std::size_t count = npos) const    {        return subview().substr(pos, count);    }    /** Return a view.        Returns a view of the whole string.        @par Exception Safety        No-throw guarantee.        @return `string_view(this->data(), this->size())`.    */    string_view    subview() const noexcept    {        return string_view( data(), size() );    }    //------------------------------------------------------    /** Copy a substring to another string.        Copies `std::min(count, size() - pos)` characters        starting at index `pos` to the string pointed        to by `dest`.        @note The resulting string is not null terminated.        @return The number of characters copied.        @param count The number of characters to copy.        @param dest The string to copy to.        @param pos The index to begin copying from. The        default argument for this parameter is `0`.        @throw `boost::system::system_error` `pos > max_size()`.    */    std::size_t    copy(        char* dest,        std::size_t count,        std::size_t pos = 0) const    {        return subview().copy(dest, count, pos);    }    //------------------------------------------------------    /** Change the size of the string.        Resizes the string to contain `count` characters.        If `count > size()`, characters with the value `0`        are appended. Otherwise, `size()` is reduced        to `count`.        @param count The size to resize the string to.        @throw `boost::system::system_error` `count > max_size()`.    */    void    resize(std::size_t count)    {        resize(count, 0);    }    /** Change the size of the string.        Resizes the string to contain `count` characters.        If `count > size()`, copies of `ch` are        appended. Otherwise, `size()` is reduced        to `count`.        @param count The size to resize the string to.        @param ch The characters to append if the size        increases.        @throw `boost::system::system_error` `count > max_size()`.    */    BOOST_JSON_DECL    void    resize(std::size_t count, char ch);    /** Increase size without changing capacity.        This increases the size of the string by `n`        characters, adjusting the position of the        terminating null for the new size. The new        characters remain uninitialized. This function        may be used to append characters directly into        the storage between `end()` and        `data() + capacity()`.        @par Precondition        @code        count <= capacity() - size()        @endcode        @param n The amount to increase the size by.    */    void    grow(std::size_t n) noexcept    {        BOOST_ASSERT(            n <= impl_.capacity() - impl_.size());        impl_.term(impl_.size() + n);    }    //------------------------------------------------------    /** Swap the contents.        Exchanges the contents of this string with another string. Ownership of        the respective `boost::container::pmr::memory_resource` objects is not        transferred.        @li If `&other == this`, do nothing. Otherwise,        @li if `*other.storage() == *this->storage()`,        ownership of the underlying memory is swapped in        constant time, with no possibility of exceptions.        All iterators and references remain valid. Otherwise,        @li the contents are logically swapped by making copies,        which can throw. In this case all iterators and        references are invalidated.        @par Complexity        Constant or linear in @ref size() plus        `other.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.    */    BOOST_JSON_DECL    void    swap(string& other);    /** Exchange the given values.        Exchanges the contents of the string `lhs` with another string `rhs`.        Ownership of the respective `boost::container::pmr::memory_resource`        objects is not transferred.        @li If `&lhs == &rhs`, do nothing. Otherwise,        @li if `*lhs.storage() == *rhs.storage()`,        ownership of the underlying memory is swapped in        constant time, with no possibility of exceptions.        All iterators and references remain valid. Otherwise,        @li the contents are logically swapped by making a copy,        which can throw. In this case all iterators and        references are invalidated.        @par Effects        @code        lhs.swap( rhs );        @endcode        @par Complexity        Constant or linear in `lhs.size() + rhs.size()`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param lhs The string to exchange.        @param rhs The string to exchange.        @see @ref string::swap    */    friend    void    swap(string& lhs, string& rhs)    {        lhs.swap(rhs);    }    //------------------------------------------------------    //    // Search    //    //------------------------------------------------------    /** Find the first occurrence of a string within the string.        Returns the lowest index `idx` greater than or equal        to `pos` where each element of `sv`  is equal to        that of `{begin() + idx, begin() + idx + sv.size())`        if one exists, and @ref npos otherwise.        @par Complexity        Linear.        @return The first occurrence of `sv` within the        string starting at the index `pos`, or @ref npos        if none exists.        @param sv The `string_view` to search for.        @param pos The index to start searching at.        The default argument for this parameter is `0`.    */    std::size_t    find(        string_view sv,        std::size_t pos = 0) const noexcept    {        return subview().find(sv, pos);    }    /** Find the first occurrence of a character within the string.        Returns the index corrosponding to the first        occurrence of `ch` within `{begin() + pos, end())`        if it exists, and @ref npos otherwise.        @par Complexity        Linear.        @return The first occurrence of `ch` within the        string starting at the index `pos`, or @ref npos        if none exists.        @param ch The character to search for.        @param pos The index to start searching at.        The default argument for this parameter is `0`.    */    std::size_t    find(        char ch,        std::size_t pos = 0) const noexcept    {        return subview().find(ch, pos);    }    //------------------------------------------------------    /** Find the last occurrence of a string within the string.        Returns the highest index `idx` less than or equal        to `pos` where each element of `sv` is equal to that        of `{begin() + idx, begin() + idx + sv.size())`        if one exists, and @ref npos otherwise.        @par Complexity        Linear.        @return The last occurrence of `sv` within the        string starting before or at the index `pos`,        or @ref npos if none exists.        @param sv The `string_view` to search for.        @param pos The index to start searching at.        The default argument for this parameter        is @ref npos.    */    std::size_t    rfind(        string_view sv,        std::size_t pos = npos) const noexcept    {        return subview().rfind(sv, pos);    }    /** Find the last occurrence of a character within the string.        Returns index corrosponding to the last occurrence        of `ch` within `{begin(), begin() + pos}` if it        exists, and @ref npos otherwise.        @par Complexity        Linear.        @return The last occurrence of `ch` within the        string starting before or at the index `pos`,        or @ref npos if none exists.        @param ch The character to search for.        @param pos The index to stop searching at.        The default argument for this parameter        is @ref npos.    */    std::size_t    rfind(        char ch,        std::size_t pos = npos) const noexcept    {        return subview().rfind(ch, pos);    }    //------------------------------------------------------    /** Find the first occurrence of any of the characters within the string.        Returns the index corrosponding to the first        occurrence of any of the characters of `sv`        within `{begin() + pos, end())` if it exists,        and @ref npos otherwise.        @par Complexity        Linear.        @return The first occurrence of any of the        characters within `sv` within the string        starting at the index `pos`, or @ref npos        if none exists.        @param sv The characters to search for.        @param pos The index to start searching at.        The default argument for this parameter is `0`.    */    std::size_t    find_first_of(        string_view sv,        std::size_t pos = 0) const noexcept    {        return subview().find_first_of(sv, pos);    }    //------------------------------------------------------    /** Find the first occurrence of any of the characters not within the string.        Returns the index corrosponding to the first        character of `{begin() + pos, end())` that is        not within `sv` if it exists, and @ref npos        otherwise.        @par Complexity        Linear.        @return The first occurrence of a character that        is not within `sv` within the string starting at        the index `pos`, or @ref npos if none exists.        @param sv The characters to ignore.        @param pos The index to start searching at.        The default argument for this parameter is `0`.    */    std::size_t    find_first_not_of(        string_view sv,        std::size_t pos = 0) const noexcept    {        return subview().find_first_not_of(sv, pos);    }    /** Find the first occurrence of a character not equal to `ch`.        Returns the index corrosponding to the first        character of `{begin() + pos, end())` that is        not equal to `ch` if it exists, and        @ref npos otherwise.        @par Complexity        Linear.        @return The first occurrence of a character that        is not equal to `ch`, or @ref npos if none exists.        @param ch The character to ignore.        @param pos The index to start searching at.        The default argument for this parameter is `0`.    */    std::size_t    find_first_not_of(        char ch,        std::size_t pos = 0) const noexcept    {        return subview().find_first_not_of(ch, pos);    }    //------------------------------------------------------    /** Find the last occurrence of any of the characters within the string.        Returns the index corrosponding to the last        occurrence of any of the characters of `sv` within        `{begin(), begin() + pos}` if it exists,        and @ref npos otherwise.        @par Complexity        Linear.        @return The last occurrence of any of the        characters within `sv` within the string starting        before or at the index `pos`, or @ref npos if        none exists.        @param sv The characters to search for.        @param pos The index to stop searching at.        The default argument for this parameter        is @ref npos.    */    std::size_t    find_last_of(        string_view sv,        std::size_t pos = npos) const noexcept    {        return subview().find_last_of(sv, pos);    }    //------------------------------------------------------    /** Find the last occurrence of a character not within the string.        Returns the index corrosponding to the last        character of `{begin(), begin() + pos}` that is not        within `sv` if it exists, and @ref npos otherwise.        @par Complexity        Linear.        @return The last occurrence of a character that is        not within `sv` within the string before or at the        index `pos`, or @ref npos if none exists.        @param sv The characters to ignore.        @param pos The index to stop searching at.        The default argument for this parameter        is @ref npos.    */    std::size_t    find_last_not_of(        string_view sv,        std::size_t pos = npos) const noexcept    {        return subview().find_last_not_of(sv, pos);    }    /** Find the last occurrence of a character not equal to `ch`.        Returns the index corrosponding to the last        character of `{begin(), begin() + pos}` that is        not equal to `ch` if it exists, and @ref npos        otherwise.        @par Complexity        Linear.        @return The last occurrence of a character that        is not equal to `ch` before or at the index `pos`,        or @ref npos if none exists.        @param ch The character to ignore.        @param pos The index to start searching at.        The default argument for this parameter        is @ref npos.    */    std::size_t    find_last_not_of(        char ch,        std::size_t pos = npos) const noexcept    {        return subview().find_last_not_of(ch, pos);    }    /** Serialize @ref string to an output stream.        This function serializes a `string` as JSON into the output stream.        @return Reference to `os`.        @par Complexity        Constant or linear in the size of `str`.        @par Exception Safety        Strong guarantee.        Calls to `memory_resource::allocate` may throw.        @param os The output stream to serialize to.        @param str The value to serialize.    */    BOOST_JSON_DECL    friend    std::ostream&    operator<<(        std::ostream& os,        string const& str);private:    class undo;    template<class It>    using iter_cat = typename        std::iterator_traits<It>::iterator_category;    template<class InputIt>    void    assign(InputIt first, InputIt last,        std::random_access_iterator_tag);    template<class InputIt>    void    assign(InputIt first, InputIt last,        std::input_iterator_tag);    template<class InputIt>    void    append(InputIt first, InputIt last,        std::random_access_iterator_tag);    template<class InputIt>    void    append(InputIt first, InputIt last,        std::input_iterator_tag);    BOOST_JSON_DECL    void    reserve_impl(std::size_t new_capacity);};//----------------------------------------------------------namespace detail{template <>inlinestring_viewto_string_view<string>(string const& s) noexcept{    return s.subview();}} // namespace detail/** Return true if lhs equals rhs.    A lexicographical comparison is used.*/#ifdef BOOST_JSON_DOCSbooloperator==(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator==(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) == detail::to_string_view(rhs);}/** Return true if lhs does not equal rhs.    A lexicographical comparison is used.*/#ifdef BOOST_JSON_DOCSbooloperator!=(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator!=(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) != detail::to_string_view(rhs);}/** Return true if lhs is less than rhs.    A lexicographical comparison is used.*/#ifdef BOOST_JSON_DOCSbooloperator<(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator<(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) < detail::to_string_view(rhs);}/** Return true if lhs is less than or equal to rhs.    A lexicographical comparison is used.*/#ifdef BOOST_JSON_DOCSbooloperator<=(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator<=(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) <= detail::to_string_view(rhs);}#ifdef BOOST_JSON_DOCSbooloperator>=(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator>=(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) >= detail::to_string_view(rhs);}/** Return true if lhs is greater than rhs.    A lexicographical comparison is used.*/#ifdef BOOST_JSON_DOCSbooloperator>(string const& lhs, string const& rhs) noexcept#elsetemplate<class T, class U>detail::string_comp_op_requirement<T, U>operator>(T const& lhs, U const& rhs) noexcept#endif{    return detail::to_string_view(lhs) > detail::to_string_view(rhs);}} // namespace json} // namespace boost// std::hash specialization#ifndef BOOST_JSON_DOCSnamespace std {template<>struct hash< ::boost::json::string >{    BOOST_JSON_DECL    std::size_t    operator()( ::boost::json::string const& js ) const noexcept;};} // std#endif#include <boost/json/impl/string.hpp>#endif
 |