123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309 |
- // Tencent is pleased to support the open source community by making RapidJSON
- // available.
- //
- // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All
- // rights reserved.
- //
- // Licensed under the MIT License (the "License"); you may not use this file
- // except in compliance with the License. You may obtain a copy of the License
- // at
- //
- // http://opensource.org/licenses/MIT
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
- // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
- // License for the specific language governing permissions and limitations under
- // the License.
- #ifndef RAPIDJSON_DOCUMENT_H_
- #define RAPIDJSON_DOCUMENT_H_
- /*! \file document.h */
- #include <limits>
- #include <new> // placement new
- #include "encodedstream.h"
- #include "internal/meta.h"
- #include "internal/strfunc.h"
- #include "memorystream.h"
- #include "reader.h"
- RAPIDJSON_DIAG_PUSH
- #ifdef __clang__
- RAPIDJSON_DIAG_OFF(padded)
- RAPIDJSON_DIAG_OFF(switch - enum)
- RAPIDJSON_DIAG_OFF(c++ 98 - compat)
- #elif defined(_MSC_VER)
- RAPIDJSON_DIAG_OFF(4127) // conditional expression is constant
- RAPIDJSON_DIAG_OFF(
- 4244) // conversion from kXxxFlags to 'uint16_t', possible loss of data
- #endif
- #ifdef __GNUC__
- RAPIDJSON_DIAG_OFF(effc++)
- #endif // __GNUC__
- #ifndef RAPIDJSON_NOMEMBERITERATORCLASS
- #include <iterator> // std::random_access_iterator_tag
- #endif
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- #include <utility> // std::move
- #endif
- RAPIDJSON_NAMESPACE_BEGIN
- // Forward declaration.
- template <typename Encoding, typename Allocator>
- class GenericValue;
- template <typename Encoding, typename Allocator, typename StackAllocator>
- class GenericDocument;
- //! Name-value pair in a JSON object value.
- /*!
- This class was internal to GenericValue. It used to be a inner struct.
- But a compiler (IBM XL C/C++ for AIX) have reported to have problem with
- that so it moved as a namespace scope struct.
- https://code.google.com/p/rapidjson/issues/detail?id=64
- */
- template <typename Encoding, typename Allocator>
- class GenericMember {
- public:
- GenericValue<Encoding, Allocator>
- name; //!< name of member (must be a string)
- GenericValue<Encoding, Allocator> value; //!< value of member.
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Move constructor in C++11
- GenericMember(GenericMember &&rhs) RAPIDJSON_NOEXCEPT
- : name(std::move(rhs.name)),
- value(std::move(rhs.value)) {}
- //! Move assignment in C++11
- GenericMember &operator=(GenericMember &&rhs) RAPIDJSON_NOEXCEPT {
- return *this = static_cast<GenericMember &>(rhs);
- }
- #endif
- //! Assignment with move semantics.
- /*! \param rhs Source of the assignment. Its name and value will become a null
- * value after assignment.
- */
- GenericMember &operator=(GenericMember &rhs) RAPIDJSON_NOEXCEPT {
- if (RAPIDJSON_LIKELY(this != &rhs)) {
- name = rhs.name;
- value = rhs.value;
- }
- return *this;
- }
- // swap() for std::sort() and other potential use in STL.
- friend inline void swap(GenericMember &a,
- GenericMember &b) RAPIDJSON_NOEXCEPT {
- a.name.Swap(b.name);
- a.value.Swap(b.value);
- }
- private:
- //! Copy constructor is not permitted.
- GenericMember(const GenericMember &rhs);
- };
- ///////////////////////////////////////////////////////////////////////////////
- // GenericMemberIterator
- #ifndef RAPIDJSON_NOMEMBERITERATORCLASS
- //! (Constant) member iterator for a JSON object value
- /*!
- \tparam Const Is this a constant iterator?
- \tparam Encoding Encoding of the value. (Even non-string values need to
- have the same encoding in a document) \tparam Allocator Allocator type for
- allocating memory of object, array and string.
- This class implements a Random Access Iterator for GenericMember elements
- of a GenericValue, see ISO/IEC 14882:2003(E) C++ standard, 24.1
- [lib.iterator.requirements].
- \note This iterator implementation is mainly intended to avoid implicit
- conversions from iterator values to \c NULL,
- e.g. from GenericValue::FindMember.
- \note Define \c RAPIDJSON_NOMEMBERITERATORCLASS to fall back to a
- pointer-based implementation, if your platform doesn't provide
- the C++ <iterator> header.
- \see GenericMember, GenericValue::MemberIterator,
- GenericValue::ConstMemberIterator
- */
- template <bool Const, typename Encoding, typename Allocator>
- class GenericMemberIterator {
- friend class GenericValue<Encoding, Allocator>;
- template <bool, typename, typename>
- friend class GenericMemberIterator;
- typedef GenericMember<Encoding, Allocator> PlainType;
- typedef typename internal::MaybeAddConst<Const, PlainType>::Type ValueType;
- public:
- //! Iterator type itself
- typedef GenericMemberIterator Iterator;
- //! Constant iterator type
- typedef GenericMemberIterator<true, Encoding, Allocator> ConstIterator;
- //! Non-constant iterator type
- typedef GenericMemberIterator<false, Encoding, Allocator> NonConstIterator;
- /** \name std::iterator_traits support */
- //@{
- typedef ValueType value_type;
- typedef ValueType *pointer;
- typedef ValueType &reference;
- typedef std::ptrdiff_t difference_type;
- typedef std::random_access_iterator_tag iterator_category;
- //@}
- //! Pointer to (const) GenericMember
- typedef pointer Pointer;
- //! Reference to (const) GenericMember
- typedef reference Reference;
- //! Signed integer type (e.g. \c ptrdiff_t)
- typedef difference_type DifferenceType;
- //! Default constructor (singular value)
- /*! Creates an iterator pointing to no element.
- \note All operations, except for comparisons, are undefined on such
- values.
- */
- GenericMemberIterator() : ptr_() {}
- //! Iterator conversions to more const
- /*!
- \param it (Non-const) iterator to copy from
- Allows the creation of an iterator from another GenericMemberIterator
- that is "less const". Especially, creating a non-constant iterator
- from a constant iterator are disabled:
- \li const -> non-const (not ok)
- \li const -> const (ok)
- \li non-const -> const (ok)
- \li non-const -> non-const (ok)
- \note If the \c Const template parameter is already \c false, this
- constructor effectively defines a regular copy-constructor.
- Otherwise, the copy constructor is implicitly defined.
- */
- GenericMemberIterator(const NonConstIterator &it) : ptr_(it.ptr_) {}
- Iterator &operator=(const NonConstIterator &it) {
- ptr_ = it.ptr_;
- return *this;
- }
- //! @name stepping
- //@{
- Iterator &operator++() {
- ++ptr_;
- return *this;
- }
- Iterator &operator--() {
- --ptr_;
- return *this;
- }
- Iterator operator++(int) {
- Iterator old(*this);
- ++ptr_;
- return old;
- }
- Iterator operator--(int) {
- Iterator old(*this);
- --ptr_;
- return old;
- }
- //@}
- //! @name increment/decrement
- //@{
- Iterator operator+(DifferenceType n) const { return Iterator(ptr_ + n); }
- Iterator operator-(DifferenceType n) const { return Iterator(ptr_ - n); }
- Iterator &operator+=(DifferenceType n) {
- ptr_ += n;
- return *this;
- }
- Iterator &operator-=(DifferenceType n) {
- ptr_ -= n;
- return *this;
- }
- //@}
- //! @name relations
- //@{
- bool operator==(ConstIterator that) const { return ptr_ == that.ptr_; }
- bool operator!=(ConstIterator that) const { return ptr_ != that.ptr_; }
- bool operator<=(ConstIterator that) const { return ptr_ <= that.ptr_; }
- bool operator>=(ConstIterator that) const { return ptr_ >= that.ptr_; }
- bool operator<(ConstIterator that) const { return ptr_ < that.ptr_; }
- bool operator>(ConstIterator that) const { return ptr_ > that.ptr_; }
- //@}
- //! @name dereference
- //@{
- Reference operator*() const { return *ptr_; }
- Pointer operator->() const { return ptr_; }
- Reference operator[](DifferenceType n) const { return ptr_[n]; }
- //@}
- //! Distance
- DifferenceType operator-(ConstIterator that) const {
- return ptr_ - that.ptr_;
- }
- private:
- //! Internal constructor from plain pointer
- explicit GenericMemberIterator(Pointer p) : ptr_(p) {}
- Pointer ptr_; //!< raw pointer
- };
- #else // RAPIDJSON_NOMEMBERITERATORCLASS
- // class-based member iterator implementation disabled, use plain pointers
- template <bool Const, typename Encoding, typename Allocator>
- class GenericMemberIterator;
- //! non-const GenericMemberIterator
- template <typename Encoding, typename Allocator>
- class GenericMemberIterator<false, Encoding, Allocator> {
- //! use plain pointer as iterator type
- typedef GenericMember<Encoding, Allocator> *Iterator;
- };
- //! const GenericMemberIterator
- template <typename Encoding, typename Allocator>
- class GenericMemberIterator<true, Encoding, Allocator> {
- //! use plain const pointer as iterator type
- typedef const GenericMember<Encoding, Allocator> *Iterator;
- };
- #endif // RAPIDJSON_NOMEMBERITERATORCLASS
- ///////////////////////////////////////////////////////////////////////////////
- // GenericStringRef
- //! Reference to a constant string (not taking a copy)
- /*!
- \tparam CharType character type of the string
- This helper class is used to automatically infer constant string
- references for string literals, especially from \c const \b (!)
- character arrays.
- The main use is for creating JSON string values without copying the
- source string via an \ref Allocator. This requires that the referenced
- string pointers have a sufficient lifetime, which exceeds the lifetime
- of the associated GenericValue.
- \b Example
- \code
- Value v("foo"); // ok, no need to copy & calculate length
- const char foo[] = "foo";
- v.SetString(foo); // ok
- const char* bar = foo;
- // Value x(bar); // not ok, can't rely on bar's lifetime
- Value x(StringRef(bar)); // lifetime explicitly guaranteed by user
- Value y(StringRef(bar, 3)); // ok, explicitly pass length
- \endcode
- \see StringRef, GenericValue::SetString
- */
- template <typename CharType>
- struct GenericStringRef {
- typedef CharType Ch; //!< character type of the string
- //! Create string reference from \c const character array
- #ifndef __clang__ // -Wdocumentation
- /*!
- This constructor implicitly creates a constant string reference from
- a \c const character array. It has better performance than
- \ref StringRef(const CharType*) by inferring the string \ref length
- from the array length, and also supports strings containing null
- characters.
-
- \tparam N length of the string, automatically inferred
-
- \param str Constant character array, lifetime assumed to be longer
- than the use of the string in e.g. a GenericValue
-
- \post \ref s == str
-
- \note Constant complexity.
- \note There is a hidden, private overload to disallow references to
- non-const character arrays to be created via this constructor.
- By this, e.g. function-scope arrays used to be filled via
- \c snprintf are excluded from consideration.
- In such cases, the referenced string should be \b copied to the
- GenericValue instead.
- */
- #endif
- template <SizeType N>
- GenericStringRef(const CharType (&str)[N]) RAPIDJSON_NOEXCEPT
- : s(str),
- length(N - 1) {}
- //! Explicitly create string reference from \c const character pointer
- #ifndef __clang__ // -Wdocumentation
- /*!
- This constructor can be used to \b explicitly create a reference to
- a constant string pointer.
-
- \see StringRef(const CharType*)
-
- \param str Constant character pointer, lifetime assumed to be longer
- than the use of the string in e.g. a GenericValue
-
- \post \ref s == str
-
- \note There is a hidden, private overload to disallow references to
- non-const character arrays to be created via this constructor.
- By this, e.g. function-scope arrays used to be filled via
- \c snprintf are excluded from consideration.
- In such cases, the referenced string should be \b copied to the
- GenericValue instead.
- */
- #endif
- explicit GenericStringRef(const CharType *str)
- : s(str), length(NotNullStrLen(str)) {}
- //! Create constant string reference from pointer and length
- #ifndef __clang__ // -Wdocumentation
- /*! \param str constant string, lifetime assumed to be longer than the use of
- the string in e.g. a GenericValue \param len length of the string,
- excluding the trailing NULL terminator
- \post \ref s == str && \ref length == len
- \note Constant complexity.
- */
- #endif
- GenericStringRef(const CharType *str, SizeType len)
- : s(RAPIDJSON_LIKELY(str) ? str : emptyString), length(len) {
- RAPIDJSON_ASSERT(str != 0 || len == 0u);
- }
- GenericStringRef(const GenericStringRef &rhs)
- : s(rhs.s), length(rhs.length) {}
- //! implicit conversion to plain CharType pointer
- operator const Ch *() const { return s; }
- const Ch *const s; //!< plain CharType pointer
- const SizeType length; //!< length of the string (excluding the trailing NULL
- //!terminator)
- private:
- SizeType NotNullStrLen(const CharType *str) {
- RAPIDJSON_ASSERT(str != 0);
- return internal::StrLen(str);
- }
- /// Empty string - used when passing in a NULL pointer
- static const Ch emptyString[];
- //! Disallow construction from non-const array
- template <SizeType N>
- GenericStringRef(CharType (&str)[N]) /* = delete */;
- //! Copy assignment operator not permitted - immutable type
- GenericStringRef &operator=(const GenericStringRef &rhs) /* = delete */;
- };
- template <typename CharType>
- const CharType GenericStringRef<CharType>::emptyString[] = {CharType()};
- //! Mark a character pointer as constant string
- /*! Mark a plain character pointer as a "string literal". This function
- can be used to avoid copying a character string to be referenced as a
- value in a JSON GenericValue object, if the string's lifetime is known
- to be valid long enough.
- \tparam CharType Character type of the string
- \param str Constant string, lifetime assumed to be longer than the use of
- the string in e.g. a GenericValue \return GenericStringRef string reference
- object \relatesalso GenericStringRef
- \see GenericValue::GenericValue(StringRefType),
- GenericValue::operator=(StringRefType),
- GenericValue::SetString(StringRefType), GenericValue::PushBack(StringRefType,
- Allocator&), GenericValue::AddMember
- */
- template <typename CharType>
- inline GenericStringRef<CharType> StringRef(const CharType *str) {
- return GenericStringRef<CharType>(str);
- }
- //! Mark a character pointer as constant string
- /*! Mark a plain character pointer as a "string literal". This function
- can be used to avoid copying a character string to be referenced as a
- value in a JSON GenericValue object, if the string's lifetime is known
- to be valid long enough.
- This version has better performance with supplied length, and also
- supports string containing null characters.
- \tparam CharType character type of the string
- \param str Constant string, lifetime assumed to be longer than the use of
- the string in e.g. a GenericValue \param length The length of source string.
- \return GenericStringRef string reference object
- \relatesalso GenericStringRef
- */
- template <typename CharType>
- inline GenericStringRef<CharType> StringRef(const CharType *str,
- size_t length) {
- return GenericStringRef<CharType>(str, SizeType(length));
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Mark a string object as constant string
- /*! Mark a string object (e.g. \c std::string) as a "string literal".
- This function can be used to avoid copying a string to be referenced as a
- value in a JSON GenericValue object, if the string's lifetime is known
- to be valid long enough.
- \tparam CharType character type of the string
- \param str Constant string, lifetime assumed to be longer than the use of
- the string in e.g. a GenericValue \return GenericStringRef string reference
- object \relatesalso GenericStringRef \note Requires the definition of the
- preprocessor symbol \ref RAPIDJSON_HAS_STDSTRING.
- */
- template <typename CharType>
- inline GenericStringRef<CharType> StringRef(
- const std::basic_string<CharType> &str) {
- return GenericStringRef<CharType>(str.data(), SizeType(str.size()));
- }
- #endif
- ///////////////////////////////////////////////////////////////////////////////
- // GenericValue type traits
- namespace internal {
- template <typename T, typename Encoding = void, typename Allocator = void>
- struct IsGenericValueImpl : FalseType {};
- // select candidates according to nested encoding and allocator types
- template <typename T>
- struct IsGenericValueImpl<T, typename Void<typename T::EncodingType>::Type,
- typename Void<typename T::AllocatorType>::Type>
- : IsBaseOf<
- GenericValue<typename T::EncodingType, typename T::AllocatorType>,
- T>::Type {};
- // helper to match arbitrary GenericValue instantiations, including derived
- // classes
- template <typename T>
- struct IsGenericValue : IsGenericValueImpl<T>::Type {};
- } // namespace internal
- ///////////////////////////////////////////////////////////////////////////////
- // TypeHelper
- namespace internal {
- template <typename ValueType, typename T>
- struct TypeHelper {};
- template <typename ValueType>
- struct TypeHelper<ValueType, bool> {
- static bool Is(const ValueType &v) { return v.IsBool(); }
- static bool Get(const ValueType &v) { return v.GetBool(); }
- static ValueType &Set(ValueType &v, bool data) { return v.SetBool(data); }
- static ValueType &Set(ValueType &v, bool data,
- typename ValueType::AllocatorType &) {
- return v.SetBool(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, int> {
- static bool Is(const ValueType &v) { return v.IsInt(); }
- static int Get(const ValueType &v) { return v.GetInt(); }
- static ValueType &Set(ValueType &v, int data) { return v.SetInt(data); }
- static ValueType &Set(ValueType &v, int data,
- typename ValueType::AllocatorType &) {
- return v.SetInt(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, unsigned> {
- static bool Is(const ValueType &v) { return v.IsUint(); }
- static unsigned Get(const ValueType &v) { return v.GetUint(); }
- static ValueType &Set(ValueType &v, unsigned data) { return v.SetUint(data); }
- static ValueType &Set(ValueType &v, unsigned data,
- typename ValueType::AllocatorType &) {
- return v.SetUint(data);
- }
- };
- #ifdef _MSC_VER
- RAPIDJSON_STATIC_ASSERT(sizeof(long) == sizeof(int));
- template <typename ValueType>
- struct TypeHelper<ValueType, long> {
- static bool Is(const ValueType &v) { return v.IsInt(); }
- static long Get(const ValueType &v) { return v.GetInt(); }
- static ValueType &Set(ValueType &v, long data) { return v.SetInt(data); }
- static ValueType &Set(ValueType &v, long data,
- typename ValueType::AllocatorType &) {
- return v.SetInt(data);
- }
- };
- RAPIDJSON_STATIC_ASSERT(sizeof(unsigned long) == sizeof(unsigned));
- template <typename ValueType>
- struct TypeHelper<ValueType, unsigned long> {
- static bool Is(const ValueType &v) { return v.IsUint(); }
- static unsigned long Get(const ValueType &v) { return v.GetUint(); }
- static ValueType &Set(ValueType &v, unsigned long data) {
- return v.SetUint(data);
- }
- static ValueType &Set(ValueType &v, unsigned long data,
- typename ValueType::AllocatorType &) {
- return v.SetUint(data);
- }
- };
- #endif
- template <typename ValueType>
- struct TypeHelper<ValueType, int64_t> {
- static bool Is(const ValueType &v) { return v.IsInt64(); }
- static int64_t Get(const ValueType &v) { return v.GetInt64(); }
- static ValueType &Set(ValueType &v, int64_t data) { return v.SetInt64(data); }
- static ValueType &Set(ValueType &v, int64_t data,
- typename ValueType::AllocatorType &) {
- return v.SetInt64(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, uint64_t> {
- static bool Is(const ValueType &v) { return v.IsUint64(); }
- static uint64_t Get(const ValueType &v) { return v.GetUint64(); }
- static ValueType &Set(ValueType &v, uint64_t data) {
- return v.SetUint64(data);
- }
- static ValueType &Set(ValueType &v, uint64_t data,
- typename ValueType::AllocatorType &) {
- return v.SetUint64(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, double> {
- static bool Is(const ValueType &v) { return v.IsDouble(); }
- static double Get(const ValueType &v) { return v.GetDouble(); }
- static ValueType &Set(ValueType &v, double data) { return v.SetDouble(data); }
- static ValueType &Set(ValueType &v, double data,
- typename ValueType::AllocatorType &) {
- return v.SetDouble(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, float> {
- static bool Is(const ValueType &v) { return v.IsFloat(); }
- static float Get(const ValueType &v) { return v.GetFloat(); }
- static ValueType &Set(ValueType &v, float data) { return v.SetFloat(data); }
- static ValueType &Set(ValueType &v, float data,
- typename ValueType::AllocatorType &) {
- return v.SetFloat(data);
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, const typename ValueType::Ch *> {
- typedef const typename ValueType::Ch *StringType;
- static bool Is(const ValueType &v) { return v.IsString(); }
- static StringType Get(const ValueType &v) { return v.GetString(); }
- static ValueType &Set(ValueType &v, const StringType data) {
- return v.SetString(typename ValueType::StringRefType(data));
- }
- static ValueType &Set(ValueType &v, const StringType data,
- typename ValueType::AllocatorType &a) {
- return v.SetString(data, a);
- }
- };
- #if RAPIDJSON_HAS_STDSTRING
- template <typename ValueType>
- struct TypeHelper<ValueType, std::basic_string<typename ValueType::Ch>> {
- typedef std::basic_string<typename ValueType::Ch> StringType;
- static bool Is(const ValueType &v) { return v.IsString(); }
- static StringType Get(const ValueType &v) {
- return StringType(v.GetString(), v.GetStringLength());
- }
- static ValueType &Set(ValueType &v, const StringType &data,
- typename ValueType::AllocatorType &a) {
- return v.SetString(data, a);
- }
- };
- #endif
- template <typename ValueType>
- struct TypeHelper<ValueType, typename ValueType::Array> {
- typedef typename ValueType::Array ArrayType;
- static bool Is(const ValueType &v) { return v.IsArray(); }
- static ArrayType Get(ValueType &v) { return v.GetArray(); }
- static ValueType &Set(ValueType &v, ArrayType data) { return v = data; }
- static ValueType &Set(ValueType &v, ArrayType data,
- typename ValueType::AllocatorType &) {
- return v = data;
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, typename ValueType::ConstArray> {
- typedef typename ValueType::ConstArray ArrayType;
- static bool Is(const ValueType &v) { return v.IsArray(); }
- static ArrayType Get(const ValueType &v) { return v.GetArray(); }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, typename ValueType::Object> {
- typedef typename ValueType::Object ObjectType;
- static bool Is(const ValueType &v) { return v.IsObject(); }
- static ObjectType Get(ValueType &v) { return v.GetObject(); }
- static ValueType &Set(ValueType &v, ObjectType data) { return v = data; }
- static ValueType &Set(ValueType &v, ObjectType data,
- typename ValueType::AllocatorType &) {
- return v = data;
- }
- };
- template <typename ValueType>
- struct TypeHelper<ValueType, typename ValueType::ConstObject> {
- typedef typename ValueType::ConstObject ObjectType;
- static bool Is(const ValueType &v) { return v.IsObject(); }
- static ObjectType Get(const ValueType &v) { return v.GetObject(); }
- };
- } // namespace internal
- // Forward declarations
- template <bool, typename>
- class GenericArray;
- template <bool, typename>
- class GenericObject;
- ///////////////////////////////////////////////////////////////////////////////
- // GenericValue
- //! Represents a JSON value. Use Value for UTF8 encoding and default allocator.
- /*!
- A JSON value can be one of 7 types. This class is a variant type supporting
- these types.
- Use the Value if UTF8 and default allocator
- \tparam Encoding Encoding of the value. (Even non-string values need to
- have the same encoding in a document) \tparam Allocator Allocator type for
- allocating memory of object, array and string.
- */
- template <typename Encoding, typename Allocator = MemoryPoolAllocator<>>
- class GenericValue {
- public:
- //! Name-value pair in an object.
- typedef GenericMember<Encoding, Allocator> Member;
- typedef Encoding EncodingType; //!< Encoding type from template parameter.
- typedef Allocator AllocatorType; //!< Allocator type from template parameter.
- typedef typename Encoding::Ch Ch; //!< Character type derived from Encoding.
- typedef GenericStringRef<Ch>
- StringRefType; //!< Reference to a constant string
- typedef typename GenericMemberIterator<false, Encoding, Allocator>::Iterator
- MemberIterator; //!< Member iterator for iterating in object.
- typedef typename GenericMemberIterator<true, Encoding, Allocator>::Iterator
- ConstMemberIterator; //!< Constant member iterator for iterating in
- //!< object.
- typedef GenericValue
- *ValueIterator; //!< Value iterator for iterating in array.
- typedef const GenericValue
- *ConstValueIterator; //!< Constant value iterator for iterating in array.
- typedef GenericValue<Encoding, Allocator>
- ValueType; //!< Value type of itself.
- typedef GenericArray<false, ValueType> Array;
- typedef GenericArray<true, ValueType> ConstArray;
- typedef GenericObject<false, ValueType> Object;
- typedef GenericObject<true, ValueType> ConstObject;
- //!@name Constructors and destructor.
- //@{
- //! Default constructor creates a null value.
- GenericValue() RAPIDJSON_NOEXCEPT : data_() { data_.f.flags = kNullFlag; }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Move constructor in C++11
- GenericValue(GenericValue &&rhs) RAPIDJSON_NOEXCEPT : data_(rhs.data_) {
- rhs.data_.f.flags = kNullFlag; // give up contents
- }
- #endif
- private:
- //! Copy constructor is not permitted.
- GenericValue(const GenericValue &rhs);
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Moving from a GenericDocument is not permitted.
- template <typename StackAllocator>
- GenericValue(GenericDocument<Encoding, Allocator, StackAllocator> &&rhs);
- //! Move assignment from a GenericDocument is not permitted.
- template <typename StackAllocator>
- GenericValue &operator=(
- GenericDocument<Encoding, Allocator, StackAllocator> &&rhs);
- #endif
- public:
- //! Constructor with JSON value type.
- /*! This creates a Value of specified type with default content.
- \param type Type of the value.
- \note Default content for number is zero.
- */
- explicit GenericValue(Type type) RAPIDJSON_NOEXCEPT : data_() {
- static const uint16_t defaultFlags[] = {
- kNullFlag, kFalseFlag, kTrueFlag, kObjectFlag,
- kArrayFlag, kShortStringFlag, kNumberAnyFlag};
- RAPIDJSON_NOEXCEPT_ASSERT(type >= kNullType && type <= kNumberType);
- data_.f.flags = defaultFlags[type];
- // Use ShortString to store empty string.
- if (type == kStringType) data_.ss.SetLength(0);
- }
- //! Explicit copy constructor (with allocator)
- /*! Creates a copy of a Value by using the given Allocator
- \tparam SourceAllocator allocator of \c rhs
- \param rhs Value to copy from (read-only)
- \param allocator Allocator for allocating copied elements and buffers.
- Commonly use GenericDocument::GetAllocator(). \param copyConstStrings Force
- copying of constant strings (e.g. referencing an in-situ buffer) \see
- CopyFrom()
- */
- template <typename SourceAllocator>
- GenericValue(const GenericValue<Encoding, SourceAllocator> &rhs,
- Allocator &allocator, bool copyConstStrings = false) {
- switch (rhs.GetType()) {
- case kObjectType: {
- SizeType count = rhs.data_.o.size;
- Member *lm = reinterpret_cast<Member *>(
- allocator.Malloc(count * sizeof(Member)));
- const typename GenericValue<Encoding, SourceAllocator>::Member *rm =
- rhs.GetMembersPointer();
- for (SizeType i = 0; i < count; i++) {
- new (&lm[i].name)
- GenericValue(rm[i].name, allocator, copyConstStrings);
- new (&lm[i].value)
- GenericValue(rm[i].value, allocator, copyConstStrings);
- }
- data_.f.flags = kObjectFlag;
- data_.o.size = data_.o.capacity = count;
- SetMembersPointer(lm);
- } break;
- case kArrayType: {
- SizeType count = rhs.data_.a.size;
- GenericValue *le = reinterpret_cast<GenericValue *>(
- allocator.Malloc(count * sizeof(GenericValue)));
- const GenericValue<Encoding, SourceAllocator> *re =
- rhs.GetElementsPointer();
- for (SizeType i = 0; i < count; i++)
- new (&le[i]) GenericValue(re[i], allocator, copyConstStrings);
- data_.f.flags = kArrayFlag;
- data_.a.size = data_.a.capacity = count;
- SetElementsPointer(le);
- } break;
- case kStringType:
- if (rhs.data_.f.flags == kConstStringFlag && !copyConstStrings) {
- data_.f.flags = rhs.data_.f.flags;
- data_ = *reinterpret_cast<const Data *>(&rhs.data_);
- } else
- SetStringRaw(StringRef(rhs.GetString(), rhs.GetStringLength()),
- allocator);
- break;
- default:
- data_.f.flags = rhs.data_.f.flags;
- data_ = *reinterpret_cast<const Data *>(&rhs.data_);
- break;
- }
- }
- //! Constructor for boolean value.
- /*! \param b Boolean value
- \note This constructor is limited to \em real boolean values and rejects
- implicitly converted types like arbitrary pointers. Use an explicit
- cast to \c bool, if you want to construct a boolean JSON value in such
- cases.
- */
- #ifndef RAPIDJSON_DOXYGEN_RUNNING // hide SFINAE from Doxygen
- template <typename T>
- explicit GenericValue(T b, RAPIDJSON_ENABLEIF((internal::IsSame<bool, T>)))
- RAPIDJSON_NOEXCEPT // See #472
- #else
- explicit GenericValue(bool b) RAPIDJSON_NOEXCEPT
- #endif
- : data_() {
- // safe-guard against failing SFINAE
- RAPIDJSON_STATIC_ASSERT((internal::IsSame<bool, T>::Value));
- data_.f.flags = b ? kTrueFlag : kFalseFlag;
- }
- //! Constructor for int value.
- explicit GenericValue(int i) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.i64 = i;
- data_.f.flags =
- (i >= 0) ? (kNumberIntFlag | kUintFlag | kUint64Flag) : kNumberIntFlag;
- }
- //! Constructor for unsigned value.
- explicit GenericValue(unsigned u) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.u64 = u;
- data_.f.flags = (u & 0x80000000)
- ? kNumberUintFlag
- : (kNumberUintFlag | kIntFlag | kInt64Flag);
- }
- //! Constructor for int64_t value.
- explicit GenericValue(int64_t i64) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.i64 = i64;
- data_.f.flags = kNumberInt64Flag;
- if (i64 >= 0) {
- data_.f.flags |= kNumberUint64Flag;
- if (!(static_cast<uint64_t>(i64) &
- RAPIDJSON_UINT64_C2(0xFFFFFFFF, 0x00000000)))
- data_.f.flags |= kUintFlag;
- if (!(static_cast<uint64_t>(i64) &
- RAPIDJSON_UINT64_C2(0xFFFFFFFF, 0x80000000)))
- data_.f.flags |= kIntFlag;
- } else if (i64 >= static_cast<int64_t>(
- RAPIDJSON_UINT64_C2(0xFFFFFFFF, 0x80000000)))
- data_.f.flags |= kIntFlag;
- }
- //! Constructor for uint64_t value.
- explicit GenericValue(uint64_t u64) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.u64 = u64;
- data_.f.flags = kNumberUint64Flag;
- if (!(u64 & RAPIDJSON_UINT64_C2(0x80000000, 0x00000000)))
- data_.f.flags |= kInt64Flag;
- if (!(u64 & RAPIDJSON_UINT64_C2(0xFFFFFFFF, 0x00000000)))
- data_.f.flags |= kUintFlag;
- if (!(u64 & RAPIDJSON_UINT64_C2(0xFFFFFFFF, 0x80000000)))
- data_.f.flags |= kIntFlag;
- }
- //! Constructor for double value.
- explicit GenericValue(double d) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.d = d;
- data_.f.flags = kNumberDoubleFlag;
- }
- //! Constructor for float value.
- explicit GenericValue(float f) RAPIDJSON_NOEXCEPT : data_() {
- data_.n.d = static_cast<double>(f);
- data_.f.flags = kNumberDoubleFlag;
- }
- //! Constructor for constant string (i.e. do not make a copy of string)
- GenericValue(const Ch *s, SizeType length) RAPIDJSON_NOEXCEPT : data_() {
- SetStringRaw(StringRef(s, length));
- }
- //! Constructor for constant string (i.e. do not make a copy of string)
- explicit GenericValue(StringRefType s) RAPIDJSON_NOEXCEPT : data_() {
- SetStringRaw(s);
- }
- //! Constructor for copy-string (i.e. do make a copy of string)
- GenericValue(const Ch *s, SizeType length, Allocator &allocator) : data_() {
- SetStringRaw(StringRef(s, length), allocator);
- }
- //! Constructor for copy-string (i.e. do make a copy of string)
- GenericValue(const Ch *s, Allocator &allocator) : data_() {
- SetStringRaw(StringRef(s), allocator);
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Constructor for copy-string from a string object (i.e. do make a copy of
- //! string)
- /*! \note Requires the definition of the preprocessor symbol \ref
- * RAPIDJSON_HAS_STDSTRING.
- */
- GenericValue(const std::basic_string<Ch> &s, Allocator &allocator) : data_() {
- SetStringRaw(StringRef(s), allocator);
- }
- #endif
- //! Constructor for Array.
- /*!
- \param a An array obtained by \c GetArray().
- \note \c Array is always pass-by-value.
- \note the source array is moved into this value and the sourec array
- becomes empty.
- */
- GenericValue(Array a) RAPIDJSON_NOEXCEPT : data_(a.value_.data_) {
- a.value_.data_ = Data();
- a.value_.data_.f.flags = kArrayFlag;
- }
- //! Constructor for Object.
- /*!
- \param o An object obtained by \c GetObject().
- \note \c Object is always pass-by-value.
- \note the source object is moved into this value and the sourec object
- becomes empty.
- */
- GenericValue(Object o) RAPIDJSON_NOEXCEPT : data_(o.value_.data_) {
- o.value_.data_ = Data();
- o.value_.data_.f.flags = kObjectFlag;
- }
- //! Destructor.
- /*! Need to destruct elements of array, members of object, or copy-string.
- */
- ~GenericValue() {
- if (Allocator::kNeedFree) { // Shortcut by Allocator's trait
- switch (data_.f.flags) {
- case kArrayFlag: {
- GenericValue *e = GetElementsPointer();
- for (GenericValue *v = e; v != e + data_.a.size; ++v)
- v->~GenericValue();
- Allocator::Free(e);
- } break;
- case kObjectFlag:
- for (MemberIterator m = MemberBegin(); m != MemberEnd(); ++m)
- m->~Member();
- Allocator::Free(GetMembersPointer());
- break;
- case kCopyStringFlag:
- Allocator::Free(const_cast<Ch *>(GetStringPointer()));
- break;
- default:
- break; // Do nothing for other types.
- }
- }
- }
- //@}
- //!@name Assignment operators
- //@{
- //! Assignment with move semantics.
- /*! \param rhs Source of the assignment. It will become a null value after
- * assignment.
- */
- GenericValue &operator=(GenericValue &rhs) RAPIDJSON_NOEXCEPT {
- if (RAPIDJSON_LIKELY(this != &rhs)) {
- this->~GenericValue();
- RawAssign(rhs);
- }
- return *this;
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Move assignment in C++11
- GenericValue &operator=(GenericValue &&rhs) RAPIDJSON_NOEXCEPT {
- return *this = rhs.Move();
- }
- #endif
- //! Assignment of constant string reference (no copy)
- /*! \param str Constant string reference to be assigned
- \note This overload is needed to avoid clashes with the generic primitive
- type assignment overload below. \see GenericStringRef, operator=(T)
- */
- GenericValue &operator=(StringRefType str) RAPIDJSON_NOEXCEPT {
- GenericValue s(str);
- return *this = s;
- }
- //! Assignment with primitive types.
- /*! \tparam T Either \ref Type, \c int, \c unsigned, \c int64_t, \c uint64_t
- \param value The value to be assigned.
- \note The source type \c T explicitly disallows all pointer types,
- especially (\c const) \ref Ch*. This helps avoiding implicitly
- referencing character strings with insufficient lifetime, use
- \ref SetString(const Ch*, Allocator&) (for copying) or
- \ref StringRef() (to explicitly mark the pointer as constant) instead.
- All other pointer types would implicitly convert to \c bool,
- use \ref SetBool() instead.
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN((internal::IsPointer<T>), (GenericValue &))
- operator=(T value) {
- GenericValue v(value);
- return *this = v;
- }
- //! Deep-copy assignment from Value
- /*! Assigns a \b copy of the Value to the current Value object
- \tparam SourceAllocator Allocator type of \c rhs
- \param rhs Value to copy from (read-only)
- \param allocator Allocator to use for copying
- \param copyConstStrings Force copying of constant strings (e.g.
- referencing an in-situ buffer)
- */
- template <typename SourceAllocator>
- GenericValue &CopyFrom(const GenericValue<Encoding, SourceAllocator> &rhs,
- Allocator &allocator, bool copyConstStrings = false) {
- RAPIDJSON_ASSERT(static_cast<void *>(this) !=
- static_cast<void const *>(&rhs));
- this->~GenericValue();
- new (this) GenericValue(rhs, allocator, copyConstStrings);
- return *this;
- }
- //! Exchange the contents of this value with those of other.
- /*!
- \param other Another value.
- \note Constant complexity.
- */
- GenericValue &Swap(GenericValue &other) RAPIDJSON_NOEXCEPT {
- GenericValue temp;
- temp.RawAssign(*this);
- RawAssign(other);
- other.RawAssign(temp);
- return *this;
- }
- //! free-standing swap function helper
- /*!
- Helper function to enable support for common swap implementation pattern
- based on \c std::swap: \code void swap(MyClass& a, MyClass& b) { using
- std::swap; swap(a.value, b.value);
- // ...
- }
- \endcode
- \see Swap()
- */
- friend inline void swap(GenericValue &a, GenericValue &b) RAPIDJSON_NOEXCEPT {
- a.Swap(b);
- }
- //! Prepare Value for move semantics
- /*! \return *this */
- GenericValue &Move() RAPIDJSON_NOEXCEPT { return *this; }
- //@}
- //!@name Equal-to and not-equal-to operators
- //@{
- //! Equal-to operator
- /*!
- \note If an object contains duplicated named member, comparing equality
- with any object is always \c false. \note Complexity is quadratic in
- Object's member number and linear for the rest (number of all values in the
- subtree and total lengths of all strings).
- */
- template <typename SourceAllocator>
- bool operator==(const GenericValue<Encoding, SourceAllocator> &rhs) const {
- typedef GenericValue<Encoding, SourceAllocator> RhsType;
- if (GetType() != rhs.GetType()) return false;
- switch (GetType()) {
- case kObjectType: // Warning: O(n^2) inner-loop
- if (data_.o.size != rhs.data_.o.size) return false;
- for (ConstMemberIterator lhsMemberItr = MemberBegin();
- lhsMemberItr != MemberEnd(); ++lhsMemberItr) {
- typename RhsType::ConstMemberIterator rhsMemberItr =
- rhs.FindMember(lhsMemberItr->name);
- if (rhsMemberItr == rhs.MemberEnd() ||
- lhsMemberItr->value != rhsMemberItr->value)
- return false;
- }
- return true;
- case kArrayType:
- if (data_.a.size != rhs.data_.a.size) return false;
- for (SizeType i = 0; i < data_.a.size; i++)
- if ((*this)[i] != rhs[i]) return false;
- return true;
- case kStringType:
- return StringEqual(rhs);
- case kNumberType:
- if (IsDouble() || rhs.IsDouble()) {
- double a = GetDouble(); // May convert from integer to double.
- double b = rhs.GetDouble(); // Ditto
- return a >= b && a <= b; // Prevent -Wfloat-equal
- } else
- return data_.n.u64 == rhs.data_.n.u64;
- default:
- return true;
- }
- }
- //! Equal-to operator with const C-string pointer
- bool operator==(const Ch *rhs) const {
- return *this == GenericValue(StringRef(rhs));
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Equal-to operator with string object
- /*! \note Requires the definition of the preprocessor symbol \ref
- * RAPIDJSON_HAS_STDSTRING.
- */
- bool operator==(const std::basic_string<Ch> &rhs) const {
- return *this == GenericValue(StringRef(rhs));
- }
- #endif
- //! Equal-to operator with primitive types
- /*! \tparam T Either \ref Type, \c int, \c unsigned, \c int64_t, \c uint64_t,
- * \c double, \c true, \c false
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (bool))
- operator==(const T &rhs) const {
- return *this == GenericValue(rhs);
- }
- //! Not-equal-to operator
- /*! \return !(*this == rhs)
- */
- template <typename SourceAllocator>
- bool operator!=(const GenericValue<Encoding, SourceAllocator> &rhs) const {
- return !(*this == rhs);
- }
- //! Not-equal-to operator with const C-string pointer
- bool operator!=(const Ch *rhs) const { return !(*this == rhs); }
- //! Not-equal-to operator with arbitrary types
- /*! \return !(*this == rhs)
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN((internal::IsGenericValue<T>), (bool))
- operator!=(const T &rhs) const {
- return !(*this == rhs);
- }
- //! Equal-to operator with arbitrary types (symmetric version)
- /*! \return (rhs == lhs)
- */
- template <typename T>
- friend RAPIDJSON_DISABLEIF_RETURN((internal::IsGenericValue<T>), (bool))
- operator==(const T &lhs, const GenericValue &rhs) {
- return rhs == lhs;
- }
- //! Not-Equal-to operator with arbitrary types (symmetric version)
- /*! \return !(rhs == lhs)
- */
- template <typename T>
- friend RAPIDJSON_DISABLEIF_RETURN((internal::IsGenericValue<T>), (bool))
- operator!=(const T &lhs, const GenericValue &rhs) {
- return !(rhs == lhs);
- }
- //@}
- //!@name Type
- //@{
- Type GetType() const { return static_cast<Type>(data_.f.flags & kTypeMask); }
- bool IsNull() const { return data_.f.flags == kNullFlag; }
- bool IsFalse() const { return data_.f.flags == kFalseFlag; }
- bool IsTrue() const { return data_.f.flags == kTrueFlag; }
- bool IsBool() const { return (data_.f.flags & kBoolFlag) != 0; }
- bool IsObject() const { return data_.f.flags == kObjectFlag; }
- bool IsArray() const { return data_.f.flags == kArrayFlag; }
- bool IsNumber() const { return (data_.f.flags & kNumberFlag) != 0; }
- bool IsInt() const { return (data_.f.flags & kIntFlag) != 0; }
- bool IsUint() const { return (data_.f.flags & kUintFlag) != 0; }
- bool IsInt64() const { return (data_.f.flags & kInt64Flag) != 0; }
- bool IsUint64() const { return (data_.f.flags & kUint64Flag) != 0; }
- bool IsDouble() const { return (data_.f.flags & kDoubleFlag) != 0; }
- bool IsString() const { return (data_.f.flags & kStringFlag) != 0; }
- // Checks whether a number can be losslessly converted to a double.
- bool IsLosslessDouble() const {
- if (!IsNumber()) return false;
- if (IsUint64()) {
- uint64_t u = GetUint64();
- volatile double d = static_cast<double>(u);
- return (d >= 0.0) &&
- (d <
- static_cast<double>((std::numeric_limits<uint64_t>::max)())) &&
- (u == static_cast<uint64_t>(d));
- }
- if (IsInt64()) {
- int64_t i = GetInt64();
- volatile double d = static_cast<double>(i);
- return (d >=
- static_cast<double>((std::numeric_limits<int64_t>::min)())) &&
- (d < static_cast<double>((std::numeric_limits<int64_t>::max)())) &&
- (i == static_cast<int64_t>(d));
- }
- return true; // double, int, uint are always lossless
- }
- // Checks whether a number is a float (possible lossy).
- bool IsFloat() const {
- if ((data_.f.flags & kDoubleFlag) == 0) return false;
- double d = GetDouble();
- return d >= -3.4028234e38 && d <= 3.4028234e38;
- }
- // Checks whether a number can be losslessly converted to a float.
- bool IsLosslessFloat() const {
- if (!IsNumber()) return false;
- double a = GetDouble();
- if (a < static_cast<double>(-(std::numeric_limits<float>::max)()) ||
- a > static_cast<double>((std::numeric_limits<float>::max)()))
- return false;
- double b = static_cast<double>(static_cast<float>(a));
- return a >= b && a <= b; // Prevent -Wfloat-equal
- }
- //@}
- //!@name Null
- //@{
- GenericValue &SetNull() {
- this->~GenericValue();
- new (this) GenericValue();
- return *this;
- }
- //@}
- //!@name Bool
- //@{
- bool GetBool() const {
- RAPIDJSON_ASSERT(IsBool());
- return data_.f.flags == kTrueFlag;
- }
- //!< Set boolean value
- /*! \post IsBool() == true */
- GenericValue &SetBool(bool b) {
- this->~GenericValue();
- new (this) GenericValue(b);
- return *this;
- }
- //@}
- //!@name Object
- //@{
- //! Set this value as an empty object.
- /*! \post IsObject() == true */
- GenericValue &SetObject() {
- this->~GenericValue();
- new (this) GenericValue(kObjectType);
- return *this;
- }
- //! Get the number of members in the object.
- SizeType MemberCount() const {
- RAPIDJSON_ASSERT(IsObject());
- return data_.o.size;
- }
- //! Get the capacity of object.
- SizeType MemberCapacity() const {
- RAPIDJSON_ASSERT(IsObject());
- return data_.o.capacity;
- }
- //! Check whether the object is empty.
- bool ObjectEmpty() const {
- RAPIDJSON_ASSERT(IsObject());
- return data_.o.size == 0;
- }
- //! Get a value from an object associated with the name.
- /*! \pre IsObject() == true
- \tparam T Either \c Ch or \c const \c Ch (template used for disambiguation
- with \ref operator[](SizeType)) \note In version 0.1x, if the member is not
- found, this function returns a null value. This makes issue 7. Since 0.2,
- if the name is not correct, it will assert. If user is unsure whether a
- member exists, user should use HasMember() first. A better approach is to
- use FindMember(). \note Linear time complexity.
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::NotExpr<
- internal::IsSame<typename internal::RemoveConst<T>::Type, Ch>>),
- (GenericValue &))
- operator[](T *name) {
- GenericValue n(StringRef(name));
- return (*this)[n];
- }
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::NotExpr<
- internal::IsSame<typename internal::RemoveConst<T>::Type, Ch>>),
- (const GenericValue &))
- operator[](T *name) const {
- return const_cast<GenericValue &>(*this)[name];
- }
- //! Get a value from an object associated with the name.
- /*! \pre IsObject() == true
- \tparam SourceAllocator Allocator of the \c name value
- \note Compared to \ref operator[](T*), this version is faster because it
- does not need a StrLen(). And it can also handle strings with embedded null
- characters.
- \note Linear time complexity.
- */
- template <typename SourceAllocator>
- GenericValue &operator[](
- const GenericValue<Encoding, SourceAllocator> &name) {
- MemberIterator member = FindMember(name);
- if (member != MemberEnd())
- return member->value;
- else {
- RAPIDJSON_ASSERT(false); // see above note
- // This will generate -Wexit-time-destructors in clang
- // static GenericValue NullValue;
- // return NullValue;
- // Use static buffer and placement-new to prevent destruction
- static char buffer[sizeof(GenericValue)];
- return *new (buffer) GenericValue();
- }
- }
- template <typename SourceAllocator>
- const GenericValue &operator[](
- const GenericValue<Encoding, SourceAllocator> &name) const {
- return const_cast<GenericValue &>(*this)[name];
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Get a value from an object associated with name (string object).
- GenericValue &operator[](const std::basic_string<Ch> &name) {
- return (*this)[GenericValue(StringRef(name))];
- }
- const GenericValue &operator[](const std::basic_string<Ch> &name) const {
- return (*this)[GenericValue(StringRef(name))];
- }
- #endif
- //! Const member iterator
- /*! \pre IsObject() == true */
- ConstMemberIterator MemberBegin() const {
- RAPIDJSON_ASSERT(IsObject());
- return ConstMemberIterator(GetMembersPointer());
- }
- //! Const \em past-the-end member iterator
- /*! \pre IsObject() == true */
- ConstMemberIterator MemberEnd() const {
- RAPIDJSON_ASSERT(IsObject());
- return ConstMemberIterator(GetMembersPointer() + data_.o.size);
- }
- //! Member iterator
- /*! \pre IsObject() == true */
- MemberIterator MemberBegin() {
- RAPIDJSON_ASSERT(IsObject());
- return MemberIterator(GetMembersPointer());
- }
- //! \em Past-the-end member iterator
- /*! \pre IsObject() == true */
- MemberIterator MemberEnd() {
- RAPIDJSON_ASSERT(IsObject());
- return MemberIterator(GetMembersPointer() + data_.o.size);
- }
- //! Request the object to have enough capacity to store members.
- /*! \param newCapacity The capacity that the object at least need to have.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \note Linear time complexity.
- */
- GenericValue &MemberReserve(SizeType newCapacity, Allocator &allocator) {
- RAPIDJSON_ASSERT(IsObject());
- if (newCapacity > data_.o.capacity) {
- SetMembersPointer(reinterpret_cast<Member *>(allocator.Realloc(
- GetMembersPointer(), data_.o.capacity * sizeof(Member),
- newCapacity * sizeof(Member))));
- data_.o.capacity = newCapacity;
- }
- return *this;
- }
- //! Check whether a member exists in the object.
- /*!
- \param name Member name to be searched.
- \pre IsObject() == true
- \return Whether a member with that name exists.
- \note It is better to use FindMember() directly if you need the obtain the
- value as well. \note Linear time complexity.
- */
- bool HasMember(const Ch *name) const {
- return FindMember(name) != MemberEnd();
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Check whether a member exists in the object with string object.
- /*!
- \param name Member name to be searched.
- \pre IsObject() == true
- \return Whether a member with that name exists.
- \note It is better to use FindMember() directly if you need the obtain the
- value as well. \note Linear time complexity.
- */
- bool HasMember(const std::basic_string<Ch> &name) const {
- return FindMember(name) != MemberEnd();
- }
- #endif
- //! Check whether a member exists in the object with GenericValue name.
- /*!
- This version is faster because it does not need a StrLen(). It can also
- handle string with null character. \param name Member name to be searched.
- \pre IsObject() == true
- \return Whether a member with that name exists.
- \note It is better to use FindMember() directly if you need the obtain the
- value as well. \note Linear time complexity.
- */
- template <typename SourceAllocator>
- bool HasMember(const GenericValue<Encoding, SourceAllocator> &name) const {
- return FindMember(name) != MemberEnd();
- }
- //! Find member by name.
- /*!
- \param name Member name to be searched.
- \pre IsObject() == true
- \return Iterator to member, if it exists.
- Otherwise returns \ref MemberEnd().
- \note Earlier versions of Rapidjson returned a \c NULL pointer, in case
- the requested member doesn't exist. For consistency with e.g.
- \c std::map, this has been changed to MemberEnd() now.
- \note Linear time complexity.
- */
- MemberIterator FindMember(const Ch *name) {
- GenericValue n(StringRef(name));
- return FindMember(n);
- }
- ConstMemberIterator FindMember(const Ch *name) const {
- return const_cast<GenericValue &>(*this).FindMember(name);
- }
- //! Find member by name.
- /*!
- This version is faster because it does not need a StrLen(). It can also
- handle string with null character. \param name Member name to be searched.
- \pre IsObject() == true
- \return Iterator to member, if it exists.
- Otherwise returns \ref MemberEnd().
- \note Earlier versions of Rapidjson returned a \c NULL pointer, in case
- the requested member doesn't exist. For consistency with e.g.
- \c std::map, this has been changed to MemberEnd() now.
- \note Linear time complexity.
- */
- template <typename SourceAllocator>
- MemberIterator FindMember(
- const GenericValue<Encoding, SourceAllocator> &name) {
- RAPIDJSON_ASSERT(IsObject());
- RAPIDJSON_ASSERT(name.IsString());
- MemberIterator member = MemberBegin();
- for (; member != MemberEnd(); ++member)
- if (name.StringEqual(member->name)) break;
- return member;
- }
- template <typename SourceAllocator>
- ConstMemberIterator FindMember(
- const GenericValue<Encoding, SourceAllocator> &name) const {
- return const_cast<GenericValue &>(*this).FindMember(name);
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Find member by string object name.
- /*!
- \param name Member name to be searched.
- \pre IsObject() == true
- \return Iterator to member, if it exists.
- Otherwise returns \ref MemberEnd().
- */
- MemberIterator FindMember(const std::basic_string<Ch> &name) {
- return FindMember(GenericValue(StringRef(name)));
- }
- ConstMemberIterator FindMember(const std::basic_string<Ch> &name) const {
- return FindMember(GenericValue(StringRef(name)));
- }
- #endif
- //! Add a member (name-value pair) to the object.
- /*! \param name A string value as name of member.
- \param value Value of any type.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \note The ownership of \c name and \c
- value will be transferred to this object on success. \pre IsObject() &&
- name.IsString() \post name.IsNull() && value.IsNull() \note Amortized
- Constant time complexity.
- */
- GenericValue &AddMember(GenericValue &name, GenericValue &value,
- Allocator &allocator) {
- RAPIDJSON_ASSERT(IsObject());
- RAPIDJSON_ASSERT(name.IsString());
- ObjectData &o = data_.o;
- if (o.size >= o.capacity)
- MemberReserve(o.capacity == 0 ? kDefaultObjectCapacity
- : (o.capacity + (o.capacity + 1) / 2),
- allocator);
- Member *members = GetMembersPointer();
- members[o.size].name.RawAssign(name);
- members[o.size].value.RawAssign(value);
- o.size++;
- return *this;
- }
- //! Add a constant string value as member (name-value pair) to the object.
- /*! \param name A string value as name of member.
- \param value constant string reference as value of member.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \pre IsObject() \note This overload is
- needed to avoid clashes with the generic primitive type
- AddMember(GenericValue&,T,Allocator&) overload below. \note Amortized
- Constant time complexity.
- */
- GenericValue &AddMember(GenericValue &name, StringRefType value,
- Allocator &allocator) {
- GenericValue v(value);
- return AddMember(name, v, allocator);
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Add a string object as member (name-value pair) to the object.
- /*! \param name A string value as name of member.
- \param value constant string reference as value of member.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \pre IsObject() \note This overload is
- needed to avoid clashes with the generic primitive type
- AddMember(GenericValue&,T,Allocator&) overload below. \note Amortized
- Constant time complexity.
- */
- GenericValue &AddMember(GenericValue &name, std::basic_string<Ch> &value,
- Allocator &allocator) {
- GenericValue v(value, allocator);
- return AddMember(name, v, allocator);
- }
- #endif
- //! Add any primitive value as member (name-value pair) to the object.
- /*! \tparam T Either \ref Type, \c int, \c unsigned, \c int64_t, \c uint64_t
- \param name A string value as name of member.
- \param value Value of primitive type \c T as value of member
- \param allocator Allocator for reallocating memory. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \pre IsObject()
- \note The source type \c T explicitly disallows all pointer types,
- especially (\c const) \ref Ch*. This helps avoiding implicitly
- referencing character strings with insufficient lifetime, use
- \ref AddMember(StringRefType, GenericValue&, Allocator&) or \ref
- AddMember(StringRefType, StringRefType, Allocator&).
- All other pointer types would implicitly convert to \c bool,
- use an explicit cast instead, if needed.
- \note Amortized Constant time complexity.
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (GenericValue &))
- AddMember(GenericValue &name, T value, Allocator &allocator) {
- GenericValue v(value);
- return AddMember(name, v, allocator);
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericValue &AddMember(GenericValue &&name, GenericValue &&value,
- Allocator &allocator) {
- return AddMember(name, value, allocator);
- }
- GenericValue &AddMember(GenericValue &&name, GenericValue &value,
- Allocator &allocator) {
- return AddMember(name, value, allocator);
- }
- GenericValue &AddMember(GenericValue &name, GenericValue &&value,
- Allocator &allocator) {
- return AddMember(name, value, allocator);
- }
- GenericValue &AddMember(StringRefType name, GenericValue &&value,
- Allocator &allocator) {
- GenericValue n(name);
- return AddMember(n, value, allocator);
- }
- #endif // RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Add a member (name-value pair) to the object.
- /*! \param name A constant string reference as name of member.
- \param value Value of any type.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \note The ownership of \c value will be
- transferred to this object on success. \pre IsObject() \post
- value.IsNull() \note Amortized Constant time complexity.
- */
- GenericValue &AddMember(StringRefType name, GenericValue &value,
- Allocator &allocator) {
- GenericValue n(name);
- return AddMember(n, value, allocator);
- }
- //! Add a constant string value as member (name-value pair) to the object.
- /*! \param name A constant string reference as name of member.
- \param value constant string reference as value of member.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \pre IsObject() \note This overload is
- needed to avoid clashes with the generic primitive type
- AddMember(StringRefType,T,Allocator&) overload below. \note Amortized
- Constant time complexity.
- */
- GenericValue &AddMember(StringRefType name, StringRefType value,
- Allocator &allocator) {
- GenericValue v(value);
- return AddMember(name, v, allocator);
- }
- //! Add any primitive value as member (name-value pair) to the object.
- /*! \tparam T Either \ref Type, \c int, \c unsigned, \c int64_t, \c uint64_t
- \param name A constant string reference as name of member.
- \param value Value of primitive type \c T as value of member
- \param allocator Allocator for reallocating memory. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \pre IsObject()
- \note The source type \c T explicitly disallows all pointer types,
- especially (\c const) \ref Ch*. This helps avoiding implicitly
- referencing character strings with insufficient lifetime, use
- \ref AddMember(StringRefType, GenericValue&, Allocator&) or \ref
- AddMember(StringRefType, StringRefType, Allocator&).
- All other pointer types would implicitly convert to \c bool,
- use an explicit cast instead, if needed.
- \note Amortized Constant time complexity.
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (GenericValue &))
- AddMember(StringRefType name, T value, Allocator &allocator) {
- GenericValue n(name);
- return AddMember(n, value, allocator);
- }
- //! Remove all members in the object.
- /*! This function do not deallocate memory in the object, i.e. the capacity is
- unchanged. \note Linear time complexity.
- */
- void RemoveAllMembers() {
- RAPIDJSON_ASSERT(IsObject());
- for (MemberIterator m = MemberBegin(); m != MemberEnd(); ++m) m->~Member();
- data_.o.size = 0;
- }
- //! Remove a member in object by its name.
- /*! \param name Name of member to be removed.
- \return Whether the member existed.
- \note This function may reorder the object members. Use \ref
- EraseMember(ConstMemberIterator) if you need to preserve the
- relative order of the remaining members.
- \note Linear time complexity.
- */
- bool RemoveMember(const Ch *name) {
- GenericValue n(StringRef(name));
- return RemoveMember(n);
- }
- #if RAPIDJSON_HAS_STDSTRING
- bool RemoveMember(const std::basic_string<Ch> &name) {
- return RemoveMember(GenericValue(StringRef(name)));
- }
- #endif
- template <typename SourceAllocator>
- bool RemoveMember(const GenericValue<Encoding, SourceAllocator> &name) {
- MemberIterator m = FindMember(name);
- if (m != MemberEnd()) {
- RemoveMember(m);
- return true;
- } else
- return false;
- }
- //! Remove a member in object by iterator.
- /*! \param m member iterator (obtained by FindMember() or MemberBegin()).
- \return the new iterator after removal.
- \note This function may reorder the object members. Use \ref
- EraseMember(ConstMemberIterator) if you need to preserve the
- relative order of the remaining members.
- \note Constant time complexity.
- */
- MemberIterator RemoveMember(MemberIterator m) {
- RAPIDJSON_ASSERT(IsObject());
- RAPIDJSON_ASSERT(data_.o.size > 0);
- RAPIDJSON_ASSERT(GetMembersPointer() != 0);
- RAPIDJSON_ASSERT(m >= MemberBegin() && m < MemberEnd());
- MemberIterator last(GetMembersPointer() + (data_.o.size - 1));
- if (data_.o.size > 1 && m != last)
- *m = *last; // Move the last one to this place
- else
- m->~Member(); // Only one left, just destroy
- --data_.o.size;
- return m;
- }
- //! Remove a member from an object by iterator.
- /*! \param pos iterator to the member to remove
- \pre IsObject() == true && \ref MemberBegin() <= \c pos < \ref MemberEnd()
- \return Iterator following the removed element.
- If the iterator \c pos refers to the last element, the \ref
- MemberEnd() iterator is returned. \note This function preserves the
- relative order of the remaining object members. If you do not need this,
- use the more efficient \ref RemoveMember(MemberIterator). \note Linear time
- complexity.
- */
- MemberIterator EraseMember(ConstMemberIterator pos) {
- return EraseMember(pos, pos + 1);
- }
- //! Remove members in the range [first, last) from an object.
- /*! \param first iterator to the first member to remove
- \param last iterator following the last member to remove
- \pre IsObject() == true && \ref MemberBegin() <= \c first <= \c last <=
- \ref MemberEnd() \return Iterator following the last removed element. \note
- This function preserves the relative order of the remaining object members.
- \note Linear time complexity.
- */
- MemberIterator EraseMember(ConstMemberIterator first,
- ConstMemberIterator last) {
- RAPIDJSON_ASSERT(IsObject());
- RAPIDJSON_ASSERT(data_.o.size > 0);
- RAPIDJSON_ASSERT(GetMembersPointer() != 0);
- RAPIDJSON_ASSERT(first >= MemberBegin());
- RAPIDJSON_ASSERT(first <= last);
- RAPIDJSON_ASSERT(last <= MemberEnd());
- MemberIterator pos = MemberBegin() + (first - MemberBegin());
- for (MemberIterator itr = pos; itr != last; ++itr) itr->~Member();
- std::memmove(static_cast<void *>(&*pos), &*last,
- static_cast<size_t>(MemberEnd() - last) * sizeof(Member));
- data_.o.size -= static_cast<SizeType>(last - first);
- return pos;
- }
- //! Erase a member in object by its name.
- /*! \param name Name of member to be removed.
- \return Whether the member existed.
- \note Linear time complexity.
- */
- bool EraseMember(const Ch *name) {
- GenericValue n(StringRef(name));
- return EraseMember(n);
- }
- #if RAPIDJSON_HAS_STDSTRING
- bool EraseMember(const std::basic_string<Ch> &name) {
- return EraseMember(GenericValue(StringRef(name)));
- }
- #endif
- template <typename SourceAllocator>
- bool EraseMember(const GenericValue<Encoding, SourceAllocator> &name) {
- MemberIterator m = FindMember(name);
- if (m != MemberEnd()) {
- EraseMember(m);
- return true;
- } else
- return false;
- }
- Object GetObject() {
- RAPIDJSON_ASSERT(IsObject());
- return Object(*this);
- }
- ConstObject GetObject() const {
- RAPIDJSON_ASSERT(IsObject());
- return ConstObject(*this);
- }
- //@}
- //!@name Array
- //@{
- //! Set this value as an empty array.
- /*! \post IsArray == true */
- GenericValue &SetArray() {
- this->~GenericValue();
- new (this) GenericValue(kArrayType);
- return *this;
- }
- //! Get the number of elements in array.
- SizeType Size() const {
- RAPIDJSON_ASSERT(IsArray());
- return data_.a.size;
- }
- //! Get the capacity of array.
- SizeType Capacity() const {
- RAPIDJSON_ASSERT(IsArray());
- return data_.a.capacity;
- }
- //! Check whether the array is empty.
- bool Empty() const {
- RAPIDJSON_ASSERT(IsArray());
- return data_.a.size == 0;
- }
- //! Remove all elements in the array.
- /*! This function do not deallocate memory in the array, i.e. the capacity is
- unchanged. \note Linear time complexity.
- */
- void Clear() {
- RAPIDJSON_ASSERT(IsArray());
- GenericValue *e = GetElementsPointer();
- for (GenericValue *v = e; v != e + data_.a.size; ++v) v->~GenericValue();
- data_.a.size = 0;
- }
- //! Get an element from array by index.
- /*! \pre IsArray() == true
- \param index Zero-based index of element.
- \see operator[](T*)
- */
- GenericValue &operator[](SizeType index) {
- RAPIDJSON_ASSERT(IsArray());
- RAPIDJSON_ASSERT(index < data_.a.size);
- return GetElementsPointer()[index];
- }
- const GenericValue &operator[](SizeType index) const {
- return const_cast<GenericValue &>(*this)[index];
- }
- //! Element iterator
- /*! \pre IsArray() == true */
- ValueIterator Begin() {
- RAPIDJSON_ASSERT(IsArray());
- return GetElementsPointer();
- }
- //! \em Past-the-end element iterator
- /*! \pre IsArray() == true */
- ValueIterator End() {
- RAPIDJSON_ASSERT(IsArray());
- return GetElementsPointer() + data_.a.size;
- }
- //! Constant element iterator
- /*! \pre IsArray() == true */
- ConstValueIterator Begin() const {
- return const_cast<GenericValue &>(*this).Begin();
- }
- //! Constant \em past-the-end element iterator
- /*! \pre IsArray() == true */
- ConstValueIterator End() const {
- return const_cast<GenericValue &>(*this).End();
- }
- //! Request the array to have enough capacity to store elements.
- /*! \param newCapacity The capacity that the array at least need to have.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \return
- The value itself for fluent API. \note Linear time complexity.
- */
- GenericValue &Reserve(SizeType newCapacity, Allocator &allocator) {
- RAPIDJSON_ASSERT(IsArray());
- if (newCapacity > data_.a.capacity) {
- SetElementsPointer(reinterpret_cast<GenericValue *>(allocator.Realloc(
- GetElementsPointer(), data_.a.capacity * sizeof(GenericValue),
- newCapacity * sizeof(GenericValue))));
- data_.a.capacity = newCapacity;
- }
- return *this;
- }
- //! Append a GenericValue at the end of the array.
- /*! \param value Value to be appended.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \pre
- IsArray() == true \post value.IsNull() == true \return The value itself for
- fluent API. \note The ownership of \c value will be transferred to this
- array on success. \note If the number of elements to be appended is known,
- calls Reserve() once first may be more efficient. \note Amortized constant
- time complexity.
- */
- GenericValue &PushBack(GenericValue &value, Allocator &allocator) {
- RAPIDJSON_ASSERT(IsArray());
- if (data_.a.size >= data_.a.capacity)
- Reserve(data_.a.capacity == 0
- ? kDefaultArrayCapacity
- : (data_.a.capacity + (data_.a.capacity + 1) / 2),
- allocator);
- GetElementsPointer()[data_.a.size++].RawAssign(value);
- return *this;
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericValue &PushBack(GenericValue &&value, Allocator &allocator) {
- return PushBack(value, allocator);
- }
- #endif // RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Append a constant string reference at the end of the array.
- /*! \param value Constant string reference to be appended.
- \param allocator Allocator for reallocating memory. It must be the same
- one used previously. Commonly use GenericDocument::GetAllocator(). \pre
- IsArray() == true \return The value itself for fluent API. \note If the
- number of elements to be appended is known, calls Reserve() once first may
- be more efficient. \note Amortized constant time complexity. \see
- GenericStringRef
- */
- GenericValue &PushBack(StringRefType value, Allocator &allocator) {
- return (*this).template PushBack<StringRefType>(value, allocator);
- }
- //! Append a primitive value at the end of the array.
- /*! \tparam T Either \ref Type, \c int, \c unsigned, \c int64_t, \c uint64_t
- \param value Value of primitive type T to be appended.
- \param allocator Allocator for reallocating memory. It must be the same
- one as used before. Commonly use GenericDocument::GetAllocator(). \pre
- IsArray() == true \return The value itself for fluent API. \note If the
- number of elements to be appended is known, calls Reserve() once first may
- be more efficient.
- \note The source type \c T explicitly disallows all pointer types,
- especially (\c const) \ref Ch*. This helps avoiding implicitly
- referencing character strings with insufficient lifetime, use
- \ref PushBack(GenericValue&, Allocator&) or \ref
- PushBack(StringRefType, Allocator&).
- All other pointer types would implicitly convert to \c bool,
- use an explicit cast instead, if needed.
- \note Amortized constant time complexity.
- */
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (GenericValue &))
- PushBack(T value, Allocator &allocator) {
- GenericValue v(value);
- return PushBack(v, allocator);
- }
- //! Remove the last element in the array.
- /*!
- \note Constant time complexity.
- */
- GenericValue &PopBack() {
- RAPIDJSON_ASSERT(IsArray());
- RAPIDJSON_ASSERT(!Empty());
- GetElementsPointer()[--data_.a.size].~GenericValue();
- return *this;
- }
- //! Remove an element of array by iterator.
- /*!
- \param pos iterator to the element to remove
- \pre IsArray() == true && \ref Begin() <= \c pos < \ref End()
- \return Iterator following the removed element. If the iterator pos refers
- to the last element, the End() iterator is returned. \note Linear time
- complexity.
- */
- ValueIterator Erase(ConstValueIterator pos) { return Erase(pos, pos + 1); }
- //! Remove elements in the range [first, last) of the array.
- /*!
- \param first iterator to the first element to remove
- \param last iterator following the last element to remove
- \pre IsArray() == true && \ref Begin() <= \c first <= \c last <= \ref
- End() \return Iterator following the last removed element. \note Linear
- time complexity.
- */
- ValueIterator Erase(ConstValueIterator first, ConstValueIterator last) {
- RAPIDJSON_ASSERT(IsArray());
- RAPIDJSON_ASSERT(data_.a.size > 0);
- RAPIDJSON_ASSERT(GetElementsPointer() != 0);
- RAPIDJSON_ASSERT(first >= Begin());
- RAPIDJSON_ASSERT(first <= last);
- RAPIDJSON_ASSERT(last <= End());
- ValueIterator pos = Begin() + (first - Begin());
- for (ValueIterator itr = pos; itr != last; ++itr) itr->~GenericValue();
- std::memmove(static_cast<void *>(pos), last,
- static_cast<size_t>(End() - last) * sizeof(GenericValue));
- data_.a.size -= static_cast<SizeType>(last - first);
- return pos;
- }
- Array GetArray() {
- RAPIDJSON_ASSERT(IsArray());
- return Array(*this);
- }
- ConstArray GetArray() const {
- RAPIDJSON_ASSERT(IsArray());
- return ConstArray(*this);
- }
- //@}
- //!@name Number
- //@{
- int GetInt() const {
- RAPIDJSON_ASSERT(data_.f.flags & kIntFlag);
- return data_.n.i.i;
- }
- unsigned GetUint() const {
- RAPIDJSON_ASSERT(data_.f.flags & kUintFlag);
- return data_.n.u.u;
- }
- int64_t GetInt64() const {
- RAPIDJSON_ASSERT(data_.f.flags & kInt64Flag);
- return data_.n.i64;
- }
- uint64_t GetUint64() const {
- RAPIDJSON_ASSERT(data_.f.flags & kUint64Flag);
- return data_.n.u64;
- }
- //! Get the value as double type.
- /*! \note If the value is 64-bit integer type, it may lose precision. Use \c
- * IsLosslessDouble() to check whether the converison is lossless.
- */
- double GetDouble() const {
- RAPIDJSON_ASSERT(IsNumber());
- if ((data_.f.flags & kDoubleFlag) != 0)
- return data_.n.d; // exact type, no conversion.
- if ((data_.f.flags & kIntFlag) != 0) return data_.n.i.i; // int -> double
- if ((data_.f.flags & kUintFlag) != 0)
- return data_.n.u.u; // unsigned -> double
- if ((data_.f.flags & kInt64Flag) != 0)
- return static_cast<double>(
- data_.n.i64); // int64_t -> double (may lose precision)
- RAPIDJSON_ASSERT((data_.f.flags & kUint64Flag) != 0);
- return static_cast<double>(
- data_.n.u64); // uint64_t -> double (may lose precision)
- }
- //! Get the value as float type.
- /*! \note If the value is 64-bit integer type, it may lose precision. Use \c
- * IsLosslessFloat() to check whether the converison is lossless.
- */
- float GetFloat() const { return static_cast<float>(GetDouble()); }
- GenericValue &SetInt(int i) {
- this->~GenericValue();
- new (this) GenericValue(i);
- return *this;
- }
- GenericValue &SetUint(unsigned u) {
- this->~GenericValue();
- new (this) GenericValue(u);
- return *this;
- }
- GenericValue &SetInt64(int64_t i64) {
- this->~GenericValue();
- new (this) GenericValue(i64);
- return *this;
- }
- GenericValue &SetUint64(uint64_t u64) {
- this->~GenericValue();
- new (this) GenericValue(u64);
- return *this;
- }
- GenericValue &SetDouble(double d) {
- this->~GenericValue();
- new (this) GenericValue(d);
- return *this;
- }
- GenericValue &SetFloat(float f) {
- this->~GenericValue();
- new (this) GenericValue(static_cast<double>(f));
- return *this;
- }
- //@}
- //!@name String
- //@{
- const Ch *GetString() const {
- RAPIDJSON_ASSERT(IsString());
- return (data_.f.flags & kInlineStrFlag) ? data_.ss.str : GetStringPointer();
- }
- //! Get the length of string.
- /*! Since rapidjson permits "\\u0000" in the json string,
- * strlen(v.GetString()) may not equal to v.GetStringLength().
- */
- SizeType GetStringLength() const {
- RAPIDJSON_ASSERT(IsString());
- return ((data_.f.flags & kInlineStrFlag) ? (data_.ss.GetLength())
- : data_.s.length);
- }
- //! Set this value as a string without copying source string.
- /*! This version has better performance with supplied length, and also support
- string containing null character. \param s source string pointer. \param
- length The length of source string, excluding the trailing null terminator.
- \return The value itself for fluent API.
- \post IsString() == true && GetString() == s && GetStringLength() ==
- length \see SetString(StringRefType)
- */
- GenericValue &SetString(const Ch *s, SizeType length) {
- return SetString(StringRef(s, length));
- }
- //! Set this value as a string without copying source string.
- /*! \param s source string reference
- \return The value itself for fluent API.
- \post IsString() == true && GetString() == s && GetStringLength() ==
- s.length
- */
- GenericValue &SetString(StringRefType s) {
- this->~GenericValue();
- SetStringRaw(s);
- return *this;
- }
- //! Set this value as a string by copying from source string.
- /*! This version has better performance with supplied length, and also support
- string containing null character. \param s source string. \param length The
- length of source string, excluding the trailing null terminator. \param
- allocator Allocator for allocating copied buffer. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \post IsString() == true && GetString() != s && strcmp(GetString(),s) == 0
- && GetStringLength() == length
- */
- GenericValue &SetString(const Ch *s, SizeType length, Allocator &allocator) {
- return SetString(StringRef(s, length), allocator);
- }
- //! Set this value as a string by copying from source string.
- /*! \param s source string.
- \param allocator Allocator for allocating copied buffer. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \post IsString() == true && GetString() != s && strcmp(GetString(),s) == 0
- && GetStringLength() == length
- */
- GenericValue &SetString(const Ch *s, Allocator &allocator) {
- return SetString(StringRef(s), allocator);
- }
- //! Set this value as a string by copying from source string.
- /*! \param s source string reference
- \param allocator Allocator for allocating copied buffer. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \post IsString() == true && GetString() != s.s && strcmp(GetString(),s) ==
- 0 && GetStringLength() == length
- */
- GenericValue &SetString(StringRefType s, Allocator &allocator) {
- this->~GenericValue();
- SetStringRaw(s, allocator);
- return *this;
- }
- #if RAPIDJSON_HAS_STDSTRING
- //! Set this value as a string by copying from source string.
- /*! \param s source string.
- \param allocator Allocator for allocating copied buffer. Commonly use
- GenericDocument::GetAllocator(). \return The value itself for fluent API.
- \post IsString() == true && GetString() != s.data() &&
- strcmp(GetString(),s.data() == 0 && GetStringLength() == s.size() \note
- Requires the definition of the preprocessor symbol \ref
- RAPIDJSON_HAS_STDSTRING.
- */
- GenericValue &SetString(const std::basic_string<Ch> &s,
- Allocator &allocator) {
- return SetString(StringRef(s), allocator);
- }
- #endif
- //@}
- //!@name Array
- //@{
- //! Templated version for checking whether this value is type T.
- /*!
- \tparam T Either \c bool, \c int, \c unsigned, \c int64_t, \c uint64_t, \c
- double, \c float, \c const \c char*, \c std::basic_string<Ch>
- */
- template <typename T>
- bool Is() const {
- return internal::TypeHelper<ValueType, T>::Is(*this);
- }
- template <typename T>
- T Get() const {
- return internal::TypeHelper<ValueType, T>::Get(*this);
- }
- template <typename T>
- T Get() {
- return internal::TypeHelper<ValueType, T>::Get(*this);
- }
- template <typename T>
- ValueType &Set(const T &data) {
- return internal::TypeHelper<ValueType, T>::Set(*this, data);
- }
- template <typename T>
- ValueType &Set(const T &data, AllocatorType &allocator) {
- return internal::TypeHelper<ValueType, T>::Set(*this, data, allocator);
- }
- //@}
- //! Generate events of this value to a Handler.
- /*! This function adopts the GoF visitor pattern.
- Typical usage is to output this JSON value as JSON text via Writer, which
- is a Handler. It can also be used to deep clone this value via
- GenericDocument, which is also a Handler. \tparam Handler type of handler.
- \param handler An object implementing concept Handler.
- */
- template <typename Handler>
- bool Accept(Handler &handler) const {
- switch (GetType()) {
- case kNullType:
- return handler.Null();
- case kFalseType:
- return handler.Bool(false);
- case kTrueType:
- return handler.Bool(true);
- case kObjectType:
- if (RAPIDJSON_UNLIKELY(!handler.StartObject())) return false;
- for (ConstMemberIterator m = MemberBegin(); m != MemberEnd(); ++m) {
- RAPIDJSON_ASSERT(m->name.IsString()); // User may change the type of
- // name by MemberIterator.
- if (RAPIDJSON_UNLIKELY(
- !handler.Key(m->name.GetString(), m->name.GetStringLength(),
- (m->name.data_.f.flags & kCopyFlag) != 0)))
- return false;
- if (RAPIDJSON_UNLIKELY(!m->value.Accept(handler))) return false;
- }
- return handler.EndObject(data_.o.size);
- case kArrayType:
- if (RAPIDJSON_UNLIKELY(!handler.StartArray())) return false;
- for (const GenericValue *v = Begin(); v != End(); ++v)
- if (RAPIDJSON_UNLIKELY(!v->Accept(handler))) return false;
- return handler.EndArray(data_.a.size);
- case kStringType:
- return handler.String(GetString(), GetStringLength(),
- (data_.f.flags & kCopyFlag) != 0);
- default:
- RAPIDJSON_ASSERT(GetType() == kNumberType);
- if (IsDouble())
- return handler.Double(data_.n.d);
- else if (IsInt())
- return handler.Int(data_.n.i.i);
- else if (IsUint())
- return handler.Uint(data_.n.u.u);
- else if (IsInt64())
- return handler.Int64(data_.n.i64);
- else
- return handler.Uint64(data_.n.u64);
- }
- }
- private:
- template <typename, typename>
- friend class GenericValue;
- template <typename, typename, typename>
- friend class GenericDocument;
- enum {
- kBoolFlag = 0x0008,
- kNumberFlag = 0x0010,
- kIntFlag = 0x0020,
- kUintFlag = 0x0040,
- kInt64Flag = 0x0080,
- kUint64Flag = 0x0100,
- kDoubleFlag = 0x0200,
- kStringFlag = 0x0400,
- kCopyFlag = 0x0800,
- kInlineStrFlag = 0x1000,
- // Initial flags of different types.
- kNullFlag = kNullType,
- kTrueFlag = kTrueType | kBoolFlag,
- kFalseFlag = kFalseType | kBoolFlag,
- kNumberIntFlag = kNumberType | kNumberFlag | kIntFlag | kInt64Flag,
- kNumberUintFlag =
- kNumberType | kNumberFlag | kUintFlag | kUint64Flag | kInt64Flag,
- kNumberInt64Flag = kNumberType | kNumberFlag | kInt64Flag,
- kNumberUint64Flag = kNumberType | kNumberFlag | kUint64Flag,
- kNumberDoubleFlag = kNumberType | kNumberFlag | kDoubleFlag,
- kNumberAnyFlag = kNumberType | kNumberFlag | kIntFlag | kInt64Flag |
- kUintFlag | kUint64Flag | kDoubleFlag,
- kConstStringFlag = kStringType | kStringFlag,
- kCopyStringFlag = kStringType | kStringFlag | kCopyFlag,
- kShortStringFlag = kStringType | kStringFlag | kCopyFlag | kInlineStrFlag,
- kObjectFlag = kObjectType,
- kArrayFlag = kArrayType,
- kTypeMask = 0x07
- };
- static const SizeType kDefaultArrayCapacity = 16;
- static const SizeType kDefaultObjectCapacity = 16;
- struct Flag {
- #if RAPIDJSON_48BITPOINTER_OPTIMIZATION
- char payload[sizeof(SizeType) * 2 +
- 6]; // 2 x SizeType + lower 48-bit pointer
- #elif RAPIDJSON_64BIT
- char payload[sizeof(SizeType) * 2 + sizeof(void *) + 6]; // 6 padding bytes
- #else
- char payload[sizeof(SizeType) * 2 + sizeof(void *) +
- 2]; // 2 padding bytes
- #endif
- uint16_t flags;
- };
- struct String {
- SizeType length;
- SizeType hashcode; //!< reserved
- const Ch *str;
- }; // 12 bytes in 32-bit mode, 16 bytes in 64-bit mode
- // implementation detail: ShortString can represent zero-terminated strings up
- // to MaxSize chars (excluding the terminating zero) and store a value to
- // determine the length of the contained string in the last character
- // str[LenPos] by storing "MaxSize - length" there. If the string to store has
- // the maximal length of MaxSize then str[LenPos] will be 0 and therefore act
- // as the string terminator as well. For getting the string length back from
- // that value just use "MaxSize - str[LenPos]". This allows to store 13-chars
- // strings in 32-bit mode, 21-chars strings in 64-bit mode, 13-chars strings
- // for RAPIDJSON_48BITPOINTER_OPTIMIZATION=1 inline (for `UTF8`-encoded
- // strings).
- struct ShortString {
- enum {
- MaxChars = sizeof(static_cast<Flag *>(0)->payload) / sizeof(Ch),
- MaxSize = MaxChars - 1,
- LenPos = MaxSize
- };
- Ch str[MaxChars];
- inline static bool Usable(SizeType len) { return (MaxSize >= len); }
- inline void SetLength(SizeType len) {
- str[LenPos] = static_cast<Ch>(MaxSize - len);
- }
- inline SizeType GetLength() const {
- return static_cast<SizeType>(MaxSize - str[LenPos]);
- }
- }; // at most as many bytes as "String" above => 12 bytes in 32-bit mode, 16
- // bytes in 64-bit mode
- // By using proper binary layout, retrieval of different integer types do not
- // need conversions.
- union Number {
- #if RAPIDJSON_ENDIAN == RAPIDJSON_LITTLEENDIAN
- struct I {
- int i;
- char padding[4];
- } i;
- struct U {
- unsigned u;
- char padding2[4];
- } u;
- #else
- struct I {
- char padding[4];
- int i;
- } i;
- struct U {
- char padding2[4];
- unsigned u;
- } u;
- #endif
- int64_t i64;
- uint64_t u64;
- double d;
- }; // 8 bytes
- struct ObjectData {
- SizeType size;
- SizeType capacity;
- Member *members;
- }; // 12 bytes in 32-bit mode, 16 bytes in 64-bit mode
- struct ArrayData {
- SizeType size;
- SizeType capacity;
- GenericValue *elements;
- }; // 12 bytes in 32-bit mode, 16 bytes in 64-bit mode
- union Data {
- String s;
- ShortString ss;
- Number n;
- ObjectData o;
- ArrayData a;
- Flag f;
- }; // 16 bytes in 32-bit mode, 24 bytes in 64-bit mode, 16 bytes in 64-bit
- // with RAPIDJSON_48BITPOINTER_OPTIMIZATION
- RAPIDJSON_FORCEINLINE const Ch *GetStringPointer() const {
- return RAPIDJSON_GETPOINTER(Ch, data_.s.str);
- }
- RAPIDJSON_FORCEINLINE const Ch *SetStringPointer(const Ch *str) {
- return RAPIDJSON_SETPOINTER(Ch, data_.s.str, str);
- }
- RAPIDJSON_FORCEINLINE GenericValue *GetElementsPointer() const {
- return RAPIDJSON_GETPOINTER(GenericValue, data_.a.elements);
- }
- RAPIDJSON_FORCEINLINE GenericValue *SetElementsPointer(
- GenericValue *elements) {
- return RAPIDJSON_SETPOINTER(GenericValue, data_.a.elements, elements);
- }
- RAPIDJSON_FORCEINLINE Member *GetMembersPointer() const {
- return RAPIDJSON_GETPOINTER(Member, data_.o.members);
- }
- RAPIDJSON_FORCEINLINE Member *SetMembersPointer(Member *members) {
- return RAPIDJSON_SETPOINTER(Member, data_.o.members, members);
- }
- // Initialize this value as array with initial data, without calling
- // destructor.
- void SetArrayRaw(GenericValue *values, SizeType count, Allocator &allocator) {
- data_.f.flags = kArrayFlag;
- if (count) {
- GenericValue *e = static_cast<GenericValue *>(
- allocator.Malloc(count * sizeof(GenericValue)));
- SetElementsPointer(e);
- std::memcpy(static_cast<void *>(e), values, count * sizeof(GenericValue));
- } else
- SetElementsPointer(0);
- data_.a.size = data_.a.capacity = count;
- }
- //! Initialize this value as object with initial data, without calling
- //! destructor.
- void SetObjectRaw(Member *members, SizeType count, Allocator &allocator) {
- data_.f.flags = kObjectFlag;
- if (count) {
- Member *m =
- static_cast<Member *>(allocator.Malloc(count * sizeof(Member)));
- SetMembersPointer(m);
- std::memcpy(static_cast<void *>(m), members, count * sizeof(Member));
- } else
- SetMembersPointer(0);
- data_.o.size = data_.o.capacity = count;
- }
- //! Initialize this value as constant string, without calling destructor.
- void SetStringRaw(StringRefType s) RAPIDJSON_NOEXCEPT {
- data_.f.flags = kConstStringFlag;
- SetStringPointer(s);
- data_.s.length = s.length;
- }
- //! Initialize this value as copy string with initial data, without calling
- //! destructor.
- void SetStringRaw(StringRefType s, Allocator &allocator) {
- Ch *str = 0;
- if (ShortString::Usable(s.length)) {
- data_.f.flags = kShortStringFlag;
- data_.ss.SetLength(s.length);
- str = data_.ss.str;
- } else {
- data_.f.flags = kCopyStringFlag;
- data_.s.length = s.length;
- str = static_cast<Ch *>(allocator.Malloc((s.length + 1) * sizeof(Ch)));
- SetStringPointer(str);
- }
- std::memcpy(str, s, s.length * sizeof(Ch));
- str[s.length] = '\0';
- }
- //! Assignment without calling destructor
- void RawAssign(GenericValue &rhs) RAPIDJSON_NOEXCEPT {
- data_ = rhs.data_;
- // data_.f.flags = rhs.data_.f.flags;
- rhs.data_.f.flags = kNullFlag;
- }
- template <typename SourceAllocator>
- bool StringEqual(const GenericValue<Encoding, SourceAllocator> &rhs) const {
- RAPIDJSON_ASSERT(IsString());
- RAPIDJSON_ASSERT(rhs.IsString());
- const SizeType len1 = GetStringLength();
- const SizeType len2 = rhs.GetStringLength();
- if (len1 != len2) {
- return false;
- }
- const Ch *const str1 = GetString();
- const Ch *const str2 = rhs.GetString();
- if (str1 == str2) {
- return true;
- } // fast path for constant string
- return (std::memcmp(str1, str2, sizeof(Ch) * len1) == 0);
- }
- Data data_;
- };
- //! GenericValue with UTF8 encoding
- typedef GenericValue<UTF8<>> Value;
- ///////////////////////////////////////////////////////////////////////////////
- // GenericDocument
- //! A document for parsing JSON text as DOM.
- /*!
- \note implements Handler concept
- \tparam Encoding Encoding for both parsing and string storage.
- \tparam Allocator Allocator for allocating memory for the DOM
- \tparam StackAllocator Allocator for allocating memory for stack during
- parsing. \warning Although GenericDocument inherits from GenericValue, the
- API does \b not provide any virtual functions, especially no virtual
- destructor. To avoid memory leaks, do not \c delete a GenericDocument object
- via a pointer to a GenericValue.
- */
- template <typename Encoding, typename Allocator = MemoryPoolAllocator<>,
- typename StackAllocator = CrtAllocator>
- class GenericDocument : public GenericValue<Encoding, Allocator> {
- public:
- typedef typename Encoding::Ch Ch; //!< Character type derived from Encoding.
- typedef GenericValue<Encoding, Allocator>
- ValueType; //!< Value type of the document.
- typedef Allocator AllocatorType; //!< Allocator type from template parameter.
- //! Constructor
- /*! Creates an empty document of specified type.
- \param type Mandatory type of object to create.
- \param allocator Optional allocator for allocating memory.
- \param stackCapacity Optional initial capacity of stack in bytes.
- \param stackAllocator Optional allocator for allocating memory for
- stack.
- */
- explicit GenericDocument(Type type, Allocator *allocator = 0,
- size_t stackCapacity = kDefaultStackCapacity,
- StackAllocator *stackAllocator = 0)
- : GenericValue<Encoding, Allocator>(type),
- allocator_(allocator),
- ownAllocator_(0),
- stack_(stackAllocator, stackCapacity),
- parseResult_() {
- if (!allocator_) ownAllocator_ = allocator_ = RAPIDJSON_NEW(Allocator)();
- }
- //! Constructor
- /*! Creates an empty document which type is Null.
- \param allocator Optional allocator for allocating memory.
- \param stackCapacity Optional initial capacity of stack in bytes.
- \param stackAllocator Optional allocator for allocating memory for
- stack.
- */
- GenericDocument(Allocator *allocator = 0,
- size_t stackCapacity = kDefaultStackCapacity,
- StackAllocator *stackAllocator = 0)
- : allocator_(allocator),
- ownAllocator_(0),
- stack_(stackAllocator, stackCapacity),
- parseResult_() {
- if (!allocator_) ownAllocator_ = allocator_ = RAPIDJSON_NEW(Allocator)();
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Move constructor in C++11
- GenericDocument(GenericDocument &&rhs) RAPIDJSON_NOEXCEPT
- : ValueType(std::forward<ValueType>(
- rhs)), // explicit cast to avoid prohibited move from Document
- allocator_(rhs.allocator_),
- ownAllocator_(rhs.ownAllocator_),
- stack_(std::move(rhs.stack_)),
- parseResult_(rhs.parseResult_) {
- rhs.allocator_ = 0;
- rhs.ownAllocator_ = 0;
- rhs.parseResult_ = ParseResult();
- }
- #endif
- ~GenericDocument() { Destroy(); }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- //! Move assignment in C++11
- GenericDocument &operator=(GenericDocument &&rhs) RAPIDJSON_NOEXCEPT {
- // The cast to ValueType is necessary here, because otherwise it would
- // attempt to call GenericValue's templated assignment operator.
- ValueType::operator=(std::forward<ValueType>(rhs));
- // Calling the destructor here would prematurely call stack_'s destructor
- Destroy();
- allocator_ = rhs.allocator_;
- ownAllocator_ = rhs.ownAllocator_;
- stack_ = std::move(rhs.stack_);
- parseResult_ = rhs.parseResult_;
- rhs.allocator_ = 0;
- rhs.ownAllocator_ = 0;
- rhs.parseResult_ = ParseResult();
- return *this;
- }
- #endif
- //! Exchange the contents of this document with those of another.
- /*!
- \param rhs Another document.
- \note Constant complexity.
- \see GenericValue::Swap
- */
- GenericDocument &Swap(GenericDocument &rhs) RAPIDJSON_NOEXCEPT {
- ValueType::Swap(rhs);
- stack_.Swap(rhs.stack_);
- internal::Swap(allocator_, rhs.allocator_);
- internal::Swap(ownAllocator_, rhs.ownAllocator_);
- internal::Swap(parseResult_, rhs.parseResult_);
- return *this;
- }
- // Allow Swap with ValueType.
- // Refer to Effective C++ 3rd Edition/Item 33: Avoid hiding inherited names.
- using ValueType::Swap;
- //! free-standing swap function helper
- /*!
- Helper function to enable support for common swap implementation pattern
- based on \c std::swap: \code void swap(MyClass& a, MyClass& b) { using
- std::swap; swap(a.doc, b.doc);
- // ...
- }
- \endcode
- \see Swap()
- */
- friend inline void swap(GenericDocument &a,
- GenericDocument &b) RAPIDJSON_NOEXCEPT {
- a.Swap(b);
- }
- //! Populate this document by a generator which produces SAX events.
- /*! \tparam Generator A functor with <tt>bool f(Handler)</tt> prototype.
- \param g Generator functor which sends SAX events to the parameter.
- \return The document itself for fluent API.
- */
- template <typename Generator>
- GenericDocument &Populate(Generator &g) {
- ClearStackOnExit scope(*this);
- if (g(*this)) {
- RAPIDJSON_ASSERT(stack_.GetSize() ==
- sizeof(ValueType)); // Got one and only one root object
- ValueType::operator=(*stack_.template Pop<ValueType>(
- 1)); // Move value from stack to document
- }
- return *this;
- }
- //!@name Parse from stream
- //!@{
- //! Parse JSON text from an input stream (with Encoding conversion)
- /*! \tparam parseFlags Combination of \ref ParseFlag.
- \tparam SourceEncoding Encoding of input stream
- \tparam InputStream Type of input stream, implementing Stream concept
- \param is Input stream to be parsed.
- \return The document itself for fluent API.
- */
- template <unsigned parseFlags, typename SourceEncoding, typename InputStream>
- GenericDocument &ParseStream(InputStream &is) {
- GenericReader<SourceEncoding, Encoding, StackAllocator> reader(
- stack_.HasAllocator() ? &stack_.GetAllocator() : 0);
- ClearStackOnExit scope(*this);
- parseResult_ = reader.template Parse<parseFlags>(is, *this);
- if (parseResult_) {
- RAPIDJSON_ASSERT(stack_.GetSize() ==
- sizeof(ValueType)); // Got one and only one root object
- ValueType::operator=(*stack_.template Pop<ValueType>(
- 1)); // Move value from stack to document
- }
- return *this;
- }
- //! Parse JSON text from an input stream
- /*! \tparam parseFlags Combination of \ref ParseFlag.
- \tparam InputStream Type of input stream, implementing Stream concept
- \param is Input stream to be parsed.
- \return The document itself for fluent API.
- */
- template <unsigned parseFlags, typename InputStream>
- GenericDocument &ParseStream(InputStream &is) {
- return ParseStream<parseFlags, Encoding, InputStream>(is);
- }
- //! Parse JSON text from an input stream (with \ref kParseDefaultFlags)
- /*! \tparam InputStream Type of input stream, implementing Stream concept
- \param is Input stream to be parsed.
- \return The document itself for fluent API.
- */
- template <typename InputStream>
- GenericDocument &ParseStream(InputStream &is) {
- return ParseStream<kParseDefaultFlags, Encoding, InputStream>(is);
- }
- //!@}
- //!@name Parse in-place from mutable string
- //!@{
- //! Parse JSON text from a mutable string
- /*! \tparam parseFlags Combination of \ref ParseFlag.
- \param str Mutable zero-terminated string to be parsed.
- \return The document itself for fluent API.
- */
- template <unsigned parseFlags>
- GenericDocument &ParseInsitu(Ch *str) {
- GenericInsituStringStream<Encoding> s(str);
- return ParseStream<parseFlags | kParseInsituFlag>(s);
- }
- //! Parse JSON text from a mutable string (with \ref kParseDefaultFlags)
- /*! \param str Mutable zero-terminated string to be parsed.
- \return The document itself for fluent API.
- */
- GenericDocument &ParseInsitu(Ch *str) {
- return ParseInsitu<kParseDefaultFlags>(str);
- }
- //!@}
- //!@name Parse from read-only string
- //!@{
- //! Parse JSON text from a read-only string (with Encoding conversion)
- /*! \tparam parseFlags Combination of \ref ParseFlag (must not contain \ref
- kParseInsituFlag). \tparam SourceEncoding Transcoding from input Encoding
- \param str Read-only zero-terminated string to be parsed.
- */
- template <unsigned parseFlags, typename SourceEncoding>
- GenericDocument &Parse(const typename SourceEncoding::Ch *str) {
- RAPIDJSON_ASSERT(!(parseFlags & kParseInsituFlag));
- GenericStringStream<SourceEncoding> s(str);
- return ParseStream<parseFlags, SourceEncoding>(s);
- }
- //! Parse JSON text from a read-only string
- /*! \tparam parseFlags Combination of \ref ParseFlag (must not contain \ref
- kParseInsituFlag). \param str Read-only zero-terminated string to be
- parsed.
- */
- template <unsigned parseFlags>
- GenericDocument &Parse(const Ch *str) {
- return Parse<parseFlags, Encoding>(str);
- }
- //! Parse JSON text from a read-only string (with \ref kParseDefaultFlags)
- /*! \param str Read-only zero-terminated string to be parsed.
- */
- GenericDocument &Parse(const Ch *str) {
- return Parse<kParseDefaultFlags>(str);
- }
- template <unsigned parseFlags, typename SourceEncoding>
- GenericDocument &Parse(const typename SourceEncoding::Ch *str,
- size_t length) {
- RAPIDJSON_ASSERT(!(parseFlags & kParseInsituFlag));
- MemoryStream ms(reinterpret_cast<const char *>(str),
- length * sizeof(typename SourceEncoding::Ch));
- EncodedInputStream<SourceEncoding, MemoryStream> is(ms);
- ParseStream<parseFlags, SourceEncoding>(is);
- return *this;
- }
- template <unsigned parseFlags>
- GenericDocument &Parse(const Ch *str, size_t length) {
- return Parse<parseFlags, Encoding>(str, length);
- }
- GenericDocument &Parse(const Ch *str, size_t length) {
- return Parse<kParseDefaultFlags>(str, length);
- }
- #if RAPIDJSON_HAS_STDSTRING
- template <unsigned parseFlags, typename SourceEncoding>
- GenericDocument &Parse(
- const std::basic_string<typename SourceEncoding::Ch> &str) {
- // c_str() is constant complexity according to standard. Should be faster
- // than Parse(const char*, size_t)
- return Parse<parseFlags, SourceEncoding>(str.c_str());
- }
- template <unsigned parseFlags>
- GenericDocument &Parse(const std::basic_string<Ch> &str) {
- return Parse<parseFlags, Encoding>(str.c_str());
- }
- GenericDocument &Parse(const std::basic_string<Ch> &str) {
- return Parse<kParseDefaultFlags>(str);
- }
- #endif // RAPIDJSON_HAS_STDSTRING
- //!@}
- //!@name Handling parse errors
- //!@{
- //! Whether a parse error has occurred in the last parsing.
- bool HasParseError() const { return parseResult_.IsError(); }
- //! Get the \ref ParseErrorCode of last parsing.
- ParseErrorCode GetParseError() const { return parseResult_.Code(); }
- //! Get the position of last parsing error in input, 0 otherwise.
- size_t GetErrorOffset() const { return parseResult_.Offset(); }
- //! Implicit conversion to get the last parse result
- #ifndef __clang // -Wdocumentation
- /*! \return \ref ParseResult of the last parse operation
- \code
- Document doc;
- ParseResult ok = doc.Parse(json);
- if (!ok)
- printf( "JSON parse error: %s (%u)\n", GetParseError_En(ok.Code()),
- ok.Offset()); \endcode
- */
- #endif
- operator ParseResult() const { return parseResult_; }
- //!@}
- //! Get the allocator of this document.
- Allocator &GetAllocator() {
- RAPIDJSON_ASSERT(allocator_);
- return *allocator_;
- }
- //! Get the capacity of stack in bytes.
- size_t GetStackCapacity() const { return stack_.GetCapacity(); }
- private:
- // clear stack on any exit from ParseStream, e.g. due to exception
- struct ClearStackOnExit {
- explicit ClearStackOnExit(GenericDocument &d) : d_(d) {}
- ~ClearStackOnExit() { d_.ClearStack(); }
- private:
- ClearStackOnExit(const ClearStackOnExit &);
- ClearStackOnExit &operator=(const ClearStackOnExit &);
- GenericDocument &d_;
- };
- // callers of the following private Handler functions
- // template <typename,typename,typename> friend class GenericReader; // for
- // parsing
- template <typename, typename>
- friend class GenericValue; // for deep copying
- public:
- // Implementation of Handler
- bool Null() {
- new (stack_.template Push<ValueType>()) ValueType();
- return true;
- }
- bool Bool(bool b) {
- new (stack_.template Push<ValueType>()) ValueType(b);
- return true;
- }
- bool Int(int i) {
- new (stack_.template Push<ValueType>()) ValueType(i);
- return true;
- }
- bool Uint(unsigned i) {
- new (stack_.template Push<ValueType>()) ValueType(i);
- return true;
- }
- bool Int64(int64_t i) {
- new (stack_.template Push<ValueType>()) ValueType(i);
- return true;
- }
- bool Uint64(uint64_t i) {
- new (stack_.template Push<ValueType>()) ValueType(i);
- return true;
- }
- bool Double(double d) {
- new (stack_.template Push<ValueType>()) ValueType(d);
- return true;
- }
- bool RawNumber(const Ch *str, SizeType length, bool copy) {
- if (copy)
- new (stack_.template Push<ValueType>())
- ValueType(str, length, GetAllocator());
- else
- new (stack_.template Push<ValueType>()) ValueType(str, length);
- return true;
- }
- bool String(const Ch *str, SizeType length, bool copy) {
- if (copy)
- new (stack_.template Push<ValueType>())
- ValueType(str, length, GetAllocator());
- else
- new (stack_.template Push<ValueType>()) ValueType(str, length);
- return true;
- }
- bool StartObject() {
- new (stack_.template Push<ValueType>()) ValueType(kObjectType);
- return true;
- }
- bool Key(const Ch *str, SizeType length, bool copy) {
- return String(str, length, copy);
- }
- bool EndObject(SizeType memberCount) {
- typename ValueType::Member *members =
- stack_.template Pop<typename ValueType::Member>(memberCount);
- stack_.template Top<ValueType>()->SetObjectRaw(members, memberCount,
- GetAllocator());
- return true;
- }
- bool StartArray() {
- new (stack_.template Push<ValueType>()) ValueType(kArrayType);
- return true;
- }
- bool EndArray(SizeType elementCount) {
- ValueType *elements = stack_.template Pop<ValueType>(elementCount);
- stack_.template Top<ValueType>()->SetArrayRaw(elements, elementCount,
- GetAllocator());
- return true;
- }
- private:
- //! Prohibit copying
- GenericDocument(const GenericDocument &);
- //! Prohibit assignment
- GenericDocument &operator=(const GenericDocument &);
- void ClearStack() {
- if (Allocator::kNeedFree)
- while (stack_.GetSize() >
- 0) // Here assumes all elements in stack array are GenericValue
- // (Member is actually 2 GenericValue objects)
- (stack_.template Pop<ValueType>(1))->~ValueType();
- else
- stack_.Clear();
- stack_.ShrinkToFit();
- }
- void Destroy() { RAPIDJSON_DELETE(ownAllocator_); }
- static const size_t kDefaultStackCapacity = 1024;
- Allocator *allocator_;
- Allocator *ownAllocator_;
- internal::Stack<StackAllocator> stack_;
- ParseResult parseResult_;
- };
- //! GenericDocument with UTF8 encoding
- typedef GenericDocument<UTF8<>> Document;
- //! Helper class for accessing Value of array type.
- /*!
- Instance of this helper class is obtained by \c GenericValue::GetArray().
- In addition to all APIs for array type, it provides range-based for loop if
- \c RAPIDJSON_HAS_CXX11_RANGE_FOR=1.
- */
- template <bool Const, typename ValueT>
- class GenericArray {
- public:
- typedef GenericArray<true, ValueT> ConstArray;
- typedef GenericArray<false, ValueT> Array;
- typedef ValueT PlainType;
- typedef typename internal::MaybeAddConst<Const, PlainType>::Type ValueType;
- typedef ValueType *ValueIterator; // This may be const or non-const iterator
- typedef const ValueT *ConstValueIterator;
- typedef typename ValueType::AllocatorType AllocatorType;
- typedef typename ValueType::StringRefType StringRefType;
- template <typename, typename>
- friend class GenericValue;
- GenericArray(const GenericArray &rhs) : value_(rhs.value_) {}
- GenericArray &operator=(const GenericArray &rhs) {
- value_ = rhs.value_;
- return *this;
- }
- ~GenericArray() {}
- SizeType Size() const { return value_.Size(); }
- SizeType Capacity() const { return value_.Capacity(); }
- bool Empty() const { return value_.Empty(); }
- void Clear() const { value_.Clear(); }
- ValueType &operator[](SizeType index) const { return value_[index]; }
- ValueIterator Begin() const { return value_.Begin(); }
- ValueIterator End() const { return value_.End(); }
- GenericArray Reserve(SizeType newCapacity, AllocatorType &allocator) const {
- value_.Reserve(newCapacity, allocator);
- return *this;
- }
- GenericArray PushBack(ValueType &value, AllocatorType &allocator) const {
- value_.PushBack(value, allocator);
- return *this;
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericArray PushBack(ValueType &&value, AllocatorType &allocator) const {
- value_.PushBack(value, allocator);
- return *this;
- }
- #endif // RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericArray PushBack(StringRefType value, AllocatorType &allocator) const {
- value_.PushBack(value, allocator);
- return *this;
- }
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (const GenericArray &))
- PushBack(T value, AllocatorType &allocator) const {
- value_.PushBack(value, allocator);
- return *this;
- }
- GenericArray PopBack() const {
- value_.PopBack();
- return *this;
- }
- ValueIterator Erase(ConstValueIterator pos) const {
- return value_.Erase(pos);
- }
- ValueIterator Erase(ConstValueIterator first, ConstValueIterator last) const {
- return value_.Erase(first, last);
- }
- #if RAPIDJSON_HAS_CXX11_RANGE_FOR
- ValueIterator begin() const { return value_.Begin(); }
- ValueIterator end() const { return value_.End(); }
- #endif
- private:
- GenericArray();
- GenericArray(ValueType &value) : value_(value) {}
- ValueType &value_;
- };
- //! Helper class for accessing Value of object type.
- /*!
- Instance of this helper class is obtained by \c GenericValue::GetObject().
- In addition to all APIs for array type, it provides range-based for loop if
- \c RAPIDJSON_HAS_CXX11_RANGE_FOR=1.
- */
- template <bool Const, typename ValueT>
- class GenericObject {
- public:
- typedef GenericObject<true, ValueT> ConstObject;
- typedef GenericObject<false, ValueT> Object;
- typedef ValueT PlainType;
- typedef typename internal::MaybeAddConst<Const, PlainType>::Type ValueType;
- typedef GenericMemberIterator<Const, typename ValueT::EncodingType,
- typename ValueT::AllocatorType>
- MemberIterator; // This may be const or non-const iterator
- typedef GenericMemberIterator<true, typename ValueT::EncodingType,
- typename ValueT::AllocatorType>
- ConstMemberIterator;
- typedef typename ValueType::AllocatorType AllocatorType;
- typedef typename ValueType::StringRefType StringRefType;
- typedef typename ValueType::EncodingType EncodingType;
- typedef typename ValueType::Ch Ch;
- template <typename, typename>
- friend class GenericValue;
- GenericObject(const GenericObject &rhs) : value_(rhs.value_) {}
- GenericObject &operator=(const GenericObject &rhs) {
- value_ = rhs.value_;
- return *this;
- }
- ~GenericObject() {}
- SizeType MemberCount() const { return value_.MemberCount(); }
- SizeType MemberCapacity() const { return value_.MemberCapacity(); }
- bool ObjectEmpty() const { return value_.ObjectEmpty(); }
- template <typename T>
- ValueType &operator[](T *name) const {
- return value_[name];
- }
- template <typename SourceAllocator>
- ValueType &operator[](
- const GenericValue<EncodingType, SourceAllocator> &name) const {
- return value_[name];
- }
- #if RAPIDJSON_HAS_STDSTRING
- ValueType &operator[](const std::basic_string<Ch> &name) const {
- return value_[name];
- }
- #endif
- MemberIterator MemberBegin() const { return value_.MemberBegin(); }
- MemberIterator MemberEnd() const { return value_.MemberEnd(); }
- GenericObject MemberReserve(SizeType newCapacity,
- AllocatorType &allocator) const {
- value_.MemberReserve(newCapacity, allocator);
- return *this;
- }
- bool HasMember(const Ch *name) const { return value_.HasMember(name); }
- #if RAPIDJSON_HAS_STDSTRING
- bool HasMember(const std::basic_string<Ch> &name) const {
- return value_.HasMember(name);
- }
- #endif
- template <typename SourceAllocator>
- bool HasMember(
- const GenericValue<EncodingType, SourceAllocator> &name) const {
- return value_.HasMember(name);
- }
- MemberIterator FindMember(const Ch *name) const {
- return value_.FindMember(name);
- }
- template <typename SourceAllocator>
- MemberIterator FindMember(
- const GenericValue<EncodingType, SourceAllocator> &name) const {
- return value_.FindMember(name);
- }
- #if RAPIDJSON_HAS_STDSTRING
- MemberIterator FindMember(const std::basic_string<Ch> &name) const {
- return value_.FindMember(name);
- }
- #endif
- GenericObject AddMember(ValueType &name, ValueType &value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- GenericObject AddMember(ValueType &name, StringRefType value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- #if RAPIDJSON_HAS_STDSTRING
- GenericObject AddMember(ValueType &name, std::basic_string<Ch> &value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- #endif
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (ValueType &))
- AddMember(ValueType &name, T value, AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericObject AddMember(ValueType &&name, ValueType &&value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- GenericObject AddMember(ValueType &&name, ValueType &value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- GenericObject AddMember(ValueType &name, ValueType &&value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- GenericObject AddMember(StringRefType name, ValueType &&value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- #endif // RAPIDJSON_HAS_CXX11_RVALUE_REFS
- GenericObject AddMember(StringRefType name, ValueType &value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- GenericObject AddMember(StringRefType name, StringRefType value,
- AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- template <typename T>
- RAPIDJSON_DISABLEIF_RETURN(
- (internal::OrExpr<internal::IsPointer<T>, internal::IsGenericValue<T>>),
- (GenericObject))
- AddMember(StringRefType name, T value, AllocatorType &allocator) const {
- value_.AddMember(name, value, allocator);
- return *this;
- }
- void RemoveAllMembers() { value_.RemoveAllMembers(); }
- bool RemoveMember(const Ch *name) const { return value_.RemoveMember(name); }
- #if RAPIDJSON_HAS_STDSTRING
- bool RemoveMember(const std::basic_string<Ch> &name) const {
- return value_.RemoveMember(name);
- }
- #endif
- template <typename SourceAllocator>
- bool RemoveMember(
- const GenericValue<EncodingType, SourceAllocator> &name) const {
- return value_.RemoveMember(name);
- }
- MemberIterator RemoveMember(MemberIterator m) const {
- return value_.RemoveMember(m);
- }
- MemberIterator EraseMember(ConstMemberIterator pos) const {
- return value_.EraseMember(pos);
- }
- MemberIterator EraseMember(ConstMemberIterator first,
- ConstMemberIterator last) const {
- return value_.EraseMember(first, last);
- }
- bool EraseMember(const Ch *name) const { return value_.EraseMember(name); }
- #if RAPIDJSON_HAS_STDSTRING
- bool EraseMember(const std::basic_string<Ch> &name) const {
- return EraseMember(ValueType(StringRef(name)));
- }
- #endif
- template <typename SourceAllocator>
- bool EraseMember(
- const GenericValue<EncodingType, SourceAllocator> &name) const {
- return value_.EraseMember(name);
- }
- #if RAPIDJSON_HAS_CXX11_RANGE_FOR
- MemberIterator begin() const { return value_.MemberBegin(); }
- MemberIterator end() const { return value_.MemberEnd(); }
- #endif
- private:
- GenericObject();
- GenericObject(ValueType &value) : value_(value) {}
- ValueType &value_;
- };
- RAPIDJSON_NAMESPACE_END
- RAPIDJSON_DIAG_POP
- #endif // RAPIDJSON_DOCUMENT_H_
|