|
@@ -1,2602 +1,2602 @@
|
|
|
-// 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.
|
|
|
+// Tencent is pleased to support the open source community by making RapidJSON available.
|
|
|
//
|
|
|
-// 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 "reader.h"
|
|
|
-#include "internal/meta.h"
|
|
|
-#include "internal/strfunc.h"
|
|
|
-#include "memorystream.h"
|
|
|
-#include "encodedstream.h"
|
|
|
-#include <new> // placement new
|
|
|
-#include <limits>
|
|
|
-
|
|
|
-RAPIDJSON_DIAG_PUSH
|
|
|
-#ifdef _MSC_VER
|
|
|
-RAPIDJSON_DIAG_OFF(4127) // conditional expression is constant
|
|
|
-RAPIDJSON_DIAG_OFF(4244) // conversion from kXxxFlags to 'uint16_t', possible loss of data
|
|
|
-#ifdef _MINWINDEF_ // see: http://stackoverflow.com/questions/22744262/cant-call-stdmax-because-minwindef-h-defines-max
|
|
|
-#ifndef NOMINMAX
|
|
|
-#pragma push_macro("min")
|
|
|
-#pragma push_macro("max")
|
|
|
-#undef min
|
|
|
-#undef max
|
|
|
-#endif
|
|
|
-#endif
|
|
|
-#endif
|
|
|
-
|
|
|
-#ifdef __clang__
|
|
|
-RAPIDJSON_DIAG_OFF(padded)
|
|
|
-RAPIDJSON_DIAG_OFF(switch-enum)
|
|
|
-RAPIDJSON_DIAG_OFF(c++98-compat)
|
|
|
-#endif
|
|
|
-
|
|
|
-#ifdef __GNUC__
|
|
|
-RAPIDJSON_DIAG_OFF(effc++)
|
|
|
-#if __GNUC__ >= 6
|
|
|
-RAPIDJSON_DIAG_OFF(terminate) // ignore throwing RAPIDJSON_ASSERT in RAPIDJSON_NOEXCEPT functions
|
|
|
-#endif
|
|
|
-#endif // __GNUC__
|
|
|
-
|
|
|
-#ifndef RAPIDJSON_NOMEMBERITERATORCLASS
|
|
|
-#include <iterator> // std::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>
|
|
|
-struct GenericMember {
|
|
|
- GenericValue<Encoding, Allocator> name; //!< name of member (must be a string)
|
|
|
- GenericValue<Encoding, Allocator> value; //!< value of member.
|
|
|
-};
|
|
|
-
|
|
|
-///////////////////////////////////////////////////////////////////////////////
|
|
|
-// 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
|
|
|
- : public std::iterator<std::random_access_iterator_tag
|
|
|
- , typename internal::MaybeAddConst<Const,GenericMember<Encoding,Allocator> >::Type> {
|
|
|
-
|
|
|
- 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;
|
|
|
- typedef std::iterator<std::random_access_iterator_tag,ValueType> BaseType;
|
|
|
-
|
|
|
-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;
|
|
|
-
|
|
|
- //! Pointer to (const) GenericMember
|
|
|
- typedef typename BaseType::pointer Pointer;
|
|
|
- //! Reference to (const) GenericMember
|
|
|
- typedef typename BaseType::reference Reference;
|
|
|
- //! Signed integer type (e.g. \c ptrdiff_t)
|
|
|
- typedef typename BaseType::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>
|
|
|
-struct GenericMemberIterator;
|
|
|
-
|
|
|
-//! non-const GenericMemberIterator
|
|
|
+// 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 "reader.h"
|
|
|
+#include "internal/meta.h"
|
|
|
+#include "internal/strfunc.h"
|
|
|
+#include "memorystream.h"
|
|
|
+#include "encodedstream.h"
|
|
|
+#include <new> // placement new
|
|
|
+#include <limits>
|
|
|
+
|
|
|
+RAPIDJSON_DIAG_PUSH
|
|
|
+#ifdef _MSC_VER
|
|
|
+RAPIDJSON_DIAG_OFF(4127) // conditional expression is constant
|
|
|
+RAPIDJSON_DIAG_OFF(4244) // conversion from kXxxFlags to 'uint16_t', possible loss of data
|
|
|
+#ifdef _MINWINDEF_ // see: http://stackoverflow.com/questions/22744262/cant-call-stdmax-because-minwindef-h-defines-max
|
|
|
+#ifndef NOMINMAX
|
|
|
+#pragma push_macro("min")
|
|
|
+#pragma push_macro("max")
|
|
|
+#undef min
|
|
|
+#undef max
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+
|
|
|
+#ifdef __clang__
|
|
|
+RAPIDJSON_DIAG_OFF(padded)
|
|
|
+RAPIDJSON_DIAG_OFF(switch-enum)
|
|
|
+RAPIDJSON_DIAG_OFF(c++98-compat)
|
|
|
+#endif
|
|
|
+
|
|
|
+#ifdef __GNUC__
|
|
|
+RAPIDJSON_DIAG_OFF(effc++)
|
|
|
+#if __GNUC__ >= 6
|
|
|
+RAPIDJSON_DIAG_OFF(terminate) // ignore throwing RAPIDJSON_ASSERT in RAPIDJSON_NOEXCEPT functions
|
|
|
+#endif
|
|
|
+#endif // __GNUC__
|
|
|
+
|
|
|
+#ifndef RAPIDJSON_NOMEMBERITERATORCLASS
|
|
|
+#include <iterator> // std::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>
|
|
|
-struct GenericMemberIterator<false,Encoding,Allocator> {
|
|
|
- //! use plain pointer as iterator type
|
|
|
- typedef GenericMember<Encoding,Allocator>* Iterator;
|
|
|
-};
|
|
|
-//! const GenericMemberIterator
|
|
|
-template <typename Encoding, typename Allocator>
|
|
|
-struct 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(internal::StrLen(str)){ RAPIDJSON_ASSERT(s != 0); }
|
|
|
-
|
|
|
- //! 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(str), length(len) { RAPIDJSON_ASSERT(s != 0); }
|
|
|
-
|
|
|
- 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:
|
|
|
- //! 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 */;
|
|
|
-};
|
|
|
-
|
|
|
-//! 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, internal::StrLen(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); }
|
|
|
-};
|
|
|
-
|
|
|
-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&) { 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[7] = {
|
|
|
- kNullFlag, kFalseFlag, kTrueFlag, kObjectFlag, kArrayFlag, kShortStringFlag,
|
|
|
- kNumberAnyFlag
|
|
|
- };
|
|
|
- RAPIDJSON_ASSERT(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().
|
|
|
- \see CopyFrom()
|
|
|
- */
|
|
|
- template <typename SourceAllocator>
|
|
|
- GenericValue(const GenericValue<Encoding,SourceAllocator>& rhs, Allocator& allocator) {
|
|
|
- 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);
|
|
|
- new (&lm[i].value) GenericValue(rm[i].value, allocator);
|
|
|
- }
|
|
|
- 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);
|
|
|
- data_.f.flags = kArrayFlag;
|
|
|
- data_.a.size = data_.a.capacity = count;
|
|
|
- SetElementsPointer(le);
|
|
|
- }
|
|
|
- break;
|
|
|
- case kStringType:
|
|
|
- if (rhs.data_.f.flags == kConstStringFlag) {
|
|
|
- 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 {
|
|
|
- RAPIDJSON_ASSERT(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
|
|
|
- */
|
|
|
- template <typename SourceAllocator>
|
|
|
- GenericValue& CopyFrom(const GenericValue<Encoding, SourceAllocator>& rhs, Allocator& allocator) {
|
|
|
- RAPIDJSON_ASSERT(static_cast<void*>(this) != static_cast<void const*>(&rhs));
|
|
|
- this->~GenericValue();
|
|
|
- new (this) GenericValue(rhs, allocator);
|
|
|
- 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 Linear time complexity (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; }
|
|
|
-
|
|
|
- //! 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); }
|
|
|
-
|
|
|
- //! 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) {
|
|
|
- if (o.capacity == 0) {
|
|
|
- o.capacity = kDefaultObjectCapacity;
|
|
|
- SetMembersPointer(reinterpret_cast<Member*>(allocator.Malloc(o.capacity * sizeof(Member))));
|
|
|
- }
|
|
|
- else {
|
|
|
- SizeType oldCapacity = o.capacity;
|
|
|
- o.capacity += (oldCapacity + 1) / 2; // grow by factor 1.5
|
|
|
- SetMembersPointer(reinterpret_cast<Member*>(allocator.Realloc(GetMembersPointer(), oldCapacity * sizeof(Member), o.capacity * sizeof(Member))));
|
|
|
- }
|
|
|
- }
|
|
|
- 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) {
|
|
|
+struct GenericMember {
|
|
|
+ GenericValue<Encoding, Allocator> name; //!< name of member (must be a string)
|
|
|
+ GenericValue<Encoding, Allocator> value; //!< value of member.
|
|
|
+};
|
|
|
+
|
|
|
+///////////////////////////////////////////////////////////////////////////////
|
|
|
+// 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
|
|
|
+ : public std::iterator<std::random_access_iterator_tag
|
|
|
+ , typename internal::MaybeAddConst<Const,GenericMember<Encoding,Allocator> >::Type> {
|
|
|
+
|
|
|
+ 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;
|
|
|
+ typedef std::iterator<std::random_access_iterator_tag,ValueType> BaseType;
|
|
|
+
|
|
|
+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;
|
|
|
+
|
|
|
+ //! Pointer to (const) GenericMember
|
|
|
+ typedef typename BaseType::pointer Pointer;
|
|
|
+ //! Reference to (const) GenericMember
|
|
|
+ typedef typename BaseType::reference Reference;
|
|
|
+ //! Signed integer type (e.g. \c ptrdiff_t)
|
|
|
+ typedef typename BaseType::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>
|
|
|
+struct GenericMemberIterator;
|
|
|
+
|
|
|
+//! non-const GenericMemberIterator
|
|
|
+template <typename Encoding, typename Allocator>
|
|
|
+struct GenericMemberIterator<false,Encoding,Allocator> {
|
|
|
+ //! use plain pointer as iterator type
|
|
|
+ typedef GenericMember<Encoding,Allocator>* Iterator;
|
|
|
+};
|
|
|
+//! const GenericMemberIterator
|
|
|
+template <typename Encoding, typename Allocator>
|
|
|
+struct 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(internal::StrLen(str)){ RAPIDJSON_ASSERT(s != 0); }
|
|
|
+
|
|
|
+ //! 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(str), length(len) { RAPIDJSON_ASSERT(s != 0); }
|
|
|
+
|
|
|
+ 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:
|
|
|
+ //! 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 */;
|
|
|
+};
|
|
|
+
|
|
|
+//! 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, internal::StrLen(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); }
|
|
|
+};
|
|
|
+
|
|
|
+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&) { 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[7] = {
|
|
|
+ kNullFlag, kFalseFlag, kTrueFlag, kObjectFlag, kArrayFlag, kShortStringFlag,
|
|
|
+ kNumberAnyFlag
|
|
|
+ };
|
|
|
+ RAPIDJSON_ASSERT(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().
|
|
|
+ \see CopyFrom()
|
|
|
+ */
|
|
|
+ template <typename SourceAllocator>
|
|
|
+ GenericValue(const GenericValue<Encoding,SourceAllocator>& rhs, Allocator& allocator) {
|
|
|
+ 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);
|
|
|
+ new (&lm[i].value) GenericValue(rm[i].value, allocator);
|
|
|
+ }
|
|
|
+ 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);
|
|
|
+ data_.f.flags = kArrayFlag;
|
|
|
+ data_.a.size = data_.a.capacity = count;
|
|
|
+ SetElementsPointer(le);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case kStringType:
|
|
|
+ if (rhs.data_.f.flags == kConstStringFlag) {
|
|
|
+ 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 {
|
|
|
+ RAPIDJSON_ASSERT(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
|
|
|
+ */
|
|
|
+ template <typename SourceAllocator>
|
|
|
+ GenericValue& CopyFrom(const GenericValue<Encoding, SourceAllocator>& rhs, Allocator& allocator) {
|
|
|
+ RAPIDJSON_ASSERT(static_cast<void*>(this) != static_cast<void const*>(&rhs));
|
|
|
+ this->~GenericValue();
|
|
|
+ new (this) GenericValue(rhs, allocator);
|
|
|
+ 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 Linear time complexity (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; }
|
|
|
+
|
|
|
+ //! 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); }
|
|
|
+
|
|
|
+ //! 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) {
|
|
|
+ if (o.capacity == 0) {
|
|
|
+ o.capacity = kDefaultObjectCapacity;
|
|
|
+ SetMembersPointer(reinterpret_cast<Member*>(allocator.Malloc(o.capacity * sizeof(Member))));
|
|
|
+ }
|
|
|
+ else {
|
|
|
+ SizeType oldCapacity = o.capacity;
|
|
|
+ o.capacity += (oldCapacity + 1) / 2; // grow by factor 1.5
|
|
|
+ SetMembersPointer(reinterpret_cast<Member*>(allocator.Realloc(GetMembersPointer(), oldCapacity * sizeof(Member), o.capacity * sizeof(Member))));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ 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());
|
|
|
- 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(&*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) {
|
|
|
+ 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(&*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());
|
|
|
- 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(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) { this->~GenericValue(); SetStringRaw(StringRef(s, length), allocator); return *this; }
|
|
|
-
|
|
|
- //! 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(s, internal::StrLen(s), allocator); }
|
|
|
-
|
|
|
-#if RAPIDJSON_HAS_STDSTRING
|
|
|
- //! Set this value as a string by copying from source string.
|
|
|
+ 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(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) { this->~GenericValue(); SetStringRaw(StringRef(s, length), allocator); return *this; }
|
|
|
+
|
|
|
+ //! 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(s.data(), SizeType(s.size()), 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(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(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;
|
|
|
- }
|
|
|
-
|
|
|
- //! 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 occured 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(); }
|
|
|
- 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(); }
|
|
|
- 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
|
|
|
-#ifdef _MINWINDEF_ // see: http://stackoverflow.com/questions/22744262/cant-call-stdmax-because-minwindef-h-defines-max
|
|
|
-#ifndef NOMINMAX
|
|
|
-#pragma pop_macro("min")
|
|
|
-#pragma pop_macro("max")
|
|
|
-#endif
|
|
|
-#endif
|
|
|
-RAPIDJSON_DIAG_POP
|
|
|
-
|
|
|
-#endif // RAPIDJSON_DOCUMENT_H_
|
|
|
+ \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(s, internal::StrLen(s), allocator); }
|
|
|
+
|
|
|
+#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(s.data(), SizeType(s.size()), 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(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(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;
|
|
|
+ }
|
|
|
+
|
|
|
+ //! 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 occured 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(); }
|
|
|
+ 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(); }
|
|
|
+ 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
|
|
|
+#ifdef _MINWINDEF_ // see: http://stackoverflow.com/questions/22744262/cant-call-stdmax-because-minwindef-h-defines-max
|
|
|
+#ifndef NOMINMAX
|
|
|
+#pragma pop_macro("min")
|
|
|
+#pragma pop_macro("max")
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+RAPIDJSON_DIAG_POP
|
|
|
+
|
|
|
+#endif // RAPIDJSON_DOCUMENT_H_
|