| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308 |
- #ifndef JSON_SHARED_STRING_H
- #define JSON_SHARED_STRING_H
- /*
- * This class allows json objects to share string
- * Since libjson is a parser, it does a lot of substrings, but since
- * a string with all of the information already exists, those substrings
- * can be infered by an offset and length and a pointer to the master
- * string
- *
- * EXPERIMENTAL, Not used yet
- */
- #include "JSONDebug.h"
- #include "JSONGlobals.h"
- #include "JSONMemory.h"
- /*
- mallocs: 3351
- frees: 3351
- reallocs: 3
- bytes: 298751 (291 KB)
- max bytes at once: 3624 (3 KB)
- avg bytes at once: 970 (0 KB)
- */
- #ifdef JSON_LESS_MEMORY
- #ifdef __GNUC__
- #pragma pack(push, 1)
- #elif _MSC_VER
- #pragma pack(push, json_shared_string_pack, 1)
- #endif
- #endif
- class json_shared_string {
- public:
- struct iterator;
- struct const_iterator {
- const_iterator(const json_char * p, const json_shared_string * pa) : parent(pa), it(p){}
-
- inline const_iterator& operator ++(void) json_nothrow { ++it; return *this; }
- inline const_iterator& operator --(void) json_nothrow { --it; return *this; }
- inline const_iterator& operator +=(long i) json_nothrow { it += i; return *this; }
- inline const_iterator& operator -=(long i) json_nothrow { it -= i; return *this; }
- inline const_iterator operator ++(int) json_nothrow {
- const_iterator result(*this);
- ++it;
- return result;
- }
- inline const_iterator operator --(int) json_nothrow {
- const_iterator result(*this);
- --it;
- return result;
- }
- inline const_iterator operator +(long i) const json_nothrow {
- const_iterator result(*this);
- result.it += i;
- return result;
- }
- inline const_iterator operator -(long i) const json_nothrow {
- const_iterator result(*this);
- result.it -= i;
- return result;
- }
- inline const json_char & operator [](size_t pos) const json_nothrow { return it[pos]; };
- inline const json_char & operator *(void) const json_nothrow { return *it; }
- inline const json_char * operator ->(void) const json_nothrow { return it; }
- inline bool operator == (const const_iterator & other) const json_nothrow { return it == other.it; }
- inline bool operator != (const const_iterator & other) const json_nothrow { return it != other.it; }
- inline bool operator > (const const_iterator & other) const json_nothrow { return it > other.it; }
- inline bool operator >= (const const_iterator & other) const json_nothrow { return it >= other.it; }
- inline bool operator < (const const_iterator & other) const json_nothrow { return it < other.it; }
- inline bool operator <= (const const_iterator & other) const json_nothrow { return it <= other.it; }
- inline bool operator == (const iterator & other) const json_nothrow { return it == other.it; }
- inline bool operator != (const iterator & other) const json_nothrow { return it != other.it; }
- inline bool operator > (const iterator & other) const json_nothrow { return it > other.it; }
- inline bool operator >= (const iterator & other) const json_nothrow { return it >= other.it; }
- inline bool operator < (const iterator & other) const json_nothrow { return it < other.it; }
- inline bool operator <= (const iterator & other) const json_nothrow { return it <= other.it; }
- inline const_iterator & operator =(const const_iterator & orig) json_nothrow { it = orig.it; return *this; }
- const_iterator (const const_iterator & orig) json_nothrow : it(orig.it) {}
- private:
- const json_shared_string * parent;
- const json_char * it;
- friend class json_shared_string;
- friend struct iterator;
- };
-
- struct iterator {
- iterator(const json_char * p, const json_shared_string * pa) : parent(pa), it(p){}
-
- inline iterator& operator ++(void) json_nothrow { ++it; return *this; }
- inline iterator& operator --(void) json_nothrow { --it; return *this; }
- inline iterator& operator +=(long i) json_nothrow { it += i; return *this; }
- inline iterator& operator -=(long i) json_nothrow { it -= i; return *this; }
- inline iterator operator ++(int) json_nothrow {
- iterator result(*this);
- ++it;
- return result;
- }
- inline iterator operator --(int) json_nothrow {
- iterator result(*this);
- --it;
- return result;
- }
- inline iterator operator +(long i) const json_nothrow {
- iterator result(*this);
- result.it += i;
- return result;
- }
- inline iterator operator -(long i) const json_nothrow {
- iterator result(*this);
- result.it -= i;
- return result;
- }
- inline const json_char & operator [](size_t pos) const json_nothrow { return it[pos]; };
- inline const json_char & operator *(void) const json_nothrow { return *it; }
- inline const json_char * operator ->(void) const json_nothrow { return it; }
- inline bool operator == (const const_iterator & other) const json_nothrow { return it == other.it; }
- inline bool operator != (const const_iterator & other) const json_nothrow { return it != other.it; }
- inline bool operator > (const const_iterator & other) const json_nothrow { return it > other.it; }
- inline bool operator >= (const const_iterator & other) const json_nothrow { return it >= other.it; }
- inline bool operator < (const const_iterator & other) const json_nothrow { return it < other.it; }
- inline bool operator <= (const const_iterator & other) const json_nothrow { return it <= other.it; }
- inline bool operator == (const iterator & other) const json_nothrow { return it == other.it; }
- inline bool operator != (const iterator & other) const json_nothrow { return it != other.it; }
- inline bool operator > (const iterator & other) const json_nothrow { return it > other.it; }
- inline bool operator >= (const iterator & other) const json_nothrow { return it >= other.it; }
- inline bool operator < (const iterator & other) const json_nothrow { return it < other.it; }
- inline bool operator <= (const iterator & other) const json_nothrow { return it <= other.it; }
- inline iterator & operator =(const iterator & orig) json_nothrow { it = orig.it; return *this; }
- iterator (const iterator & orig) json_nothrow : it(orig.it) {}
- private:
- const json_shared_string * parent;
- const json_char * it;
- friend class json_shared_string;
- friend struct const_iterator;
- };
- inline json_shared_string::iterator begin(void){
- iterator res = iterator(data(), this);
- return res;
- }
- inline json_shared_string::iterator end(void){
- iterator res = iterator(data() + len, this);
- return res;
- }
- inline json_shared_string::const_iterator begin(void) const {
- const_iterator res = const_iterator(data(), this);
- return res;
- }
- inline json_shared_string::const_iterator end(void) const {
- const_iterator res = const_iterator(data() + len, this);
- return res;
- }
-
- inline json_string::iterator std_begin(void){
- return _str -> mystring.begin() + offset;
- }
- inline json_string::iterator std_end(void){
- return std_begin() + len;
- }
-
- inline json_string::const_iterator std_begin(void) const{
- return _str -> mystring.begin() + offset;
- }
- inline json_string::const_iterator std_end(void) const{
- return std_begin() + len;
- }
-
- inline json_shared_string(void) : offset(0), len(0), _str(new(json_malloc<json_shared_string_internal>(1)) json_shared_string_internal(json_global(EMPTY_JSON_STRING))) {}
-
- inline json_shared_string(const json_string & str) : offset(0), len(str.length()), _str(new(json_malloc<json_shared_string_internal>(1)) json_shared_string_internal(str)) {}
-
- inline json_shared_string(const json_shared_string & str, size_t _offset, size_t _len) : _str(str._str), offset(str.offset + _offset), len(_len) {
- ++_str -> refCount;
- }
-
- inline json_shared_string(const json_shared_string & str, size_t _offset) : _str(str._str), offset(str.offset + _offset), len(str.len - _offset) {
- ++_str -> refCount;
- }
-
- inline json_shared_string(const iterator & s, const iterator & e) : _str(s.parent -> _str), offset(s.it - s.parent -> _str -> mystring.data()), len(e.it - s.it){
- ++_str -> refCount;
- }
-
- inline ~json_shared_string(void){
- deref();
- }
-
- inline bool empty(void) const { return len == 0; }
-
- size_t find(json_char ch, size_t pos = 0) const {
- if (_str -> refCount == 1) return _str -> mystring.find(ch, pos);
- json_string::const_iterator e = std_end();
- for(json_string::const_iterator b = std_begin() + pos; b != e; ++b){
- if (*b == ch) return b - std_begin();
- }
- return json_string::npos;
- }
-
- inline json_char & operator[] (size_t loc){
- return _str -> mystring[loc + offset];
- }
- inline json_char operator[] (size_t loc) const {
- return _str -> mystring[loc + offset];
- }
- inline void clear(){ len = 0; }
- inline size_t length() const { return len; }
- inline const json_char * c_str() const { return toString().c_str(); }
- inline const json_char * data() const { return _str -> mystring.data() + offset; }
-
- inline bool operator != (const json_shared_string & other) const {
- if ((other._str == _str) && (other.len == len) && (other.offset == offset)) return false;
- return other.toString() != toString();
- }
-
- inline bool operator == (const json_shared_string & other) const {
- if ((other._str == _str) && (other.len == len) && (other.offset == offset)) return true;
- return other.toString() == toString();
- }
-
- inline bool operator == (const json_string & other) const {
- return other == toString();
- }
-
- json_string & toString(void) const {
- //gonna have to do a real substring now anyway, so do it completely
- if (_str -> refCount == 1){
- if (offset || len != _str -> mystring.length()){
- _str -> mystring = json_string(std_begin(), std_end());
- }
- } else if (offset || len != _str -> mystring.length()){
- --_str -> refCount; //dont use deref because I know its not going to be deleted
- _str = new(json_malloc<json_shared_string_internal>(1)) json_shared_string_internal(json_string(std_begin(), std_end()));
- }
- offset = 0;
- return _str -> mystring;
- }
-
-
- inline void assign(const json_shared_string & other, size_t _offset, size_t _len){
- if (other._str != _str){
- deref();
- _str = other._str;
- }
- ++_str -> refCount;
- offset = other.offset + _offset;
- len = _len;
- }
-
- json_shared_string(const json_shared_string & other) : _str(other._str), offset(other.offset), len(other.len){
- ++_str -> refCount;
- }
-
- json_shared_string & operator =(const json_shared_string & other){
- if (other._str != _str){
- deref();
- _str = other._str;
- ++_str -> refCount;
- }
- offset = other.offset;
- len = other.len;
- return *this;
- }
-
- json_shared_string & operator += (const json_char c){
- toString() += c;
- ++len;
- return *this;
- }
-
- //when doing a plus equal of another string, see if it shares the string and starts where this one left off, in which case just increase len
- JSON_PRIVATE
- struct json_shared_string_internal {
- inline json_shared_string_internal(const json_string & _mystring) : mystring(_mystring), refCount(1) {}
- json_string mystring;
- size_t refCount PACKED(20);
- };
- inline void deref(void){
- if (--_str -> refCount == 0){
- _str -> ~json_shared_string_internal();
- libjson_free<json_shared_string_internal>(_str);
- }
- }
- mutable json_shared_string_internal * _str;
- mutable size_t offset PACKED(20);
- mutable size_t len PACKED(20);
- };
- #ifdef JSON_LESS_MEMORY
- #ifdef __GNUC__
- #pragma pack(pop)
- #elif _MSC_VER
- #pragma pack(pop, json_shared_string_pack,)
- #endif
- #endif
- #endif
|