Sun, 04 May 2014 18:31:40 +0300
- refactor the string class
src/string.cpp | file | annotate | diff | comparison | revisions | |
src/string.h | file | annotate | diff | comparison | revisions |
--- a/src/string.cpp Fri May 02 21:42:11 2014 +0300 +++ b/src/string.cpp Sun May 04 18:31:40 2014 +0300 @@ -34,7 +34,7 @@ // int String::compare (const String& other) const { - return m_string.compare (other.stdString()); + return _string.compare (other.stdString()); } // ============================================================================= @@ -42,27 +42,23 @@ void String::trim (int n) { if (n > 0) - m_string = mid (0, length() - n).stdString(); + _string = mid (0, length() - n).stdString(); else - m_string = mid (n, -1).stdString(); + _string = mid (n, -1).stdString(); } // ============================================================================= // -String String::strip (const List< char >& unwanted) +String String::strip (const List<char>& unwanted) { - String copy (m_string); + String copy (_string); for (char c : unwanted) - for (int i = 0; i < copy.length(); ++i) - if (copy[i] == c) - copy.removeAt (i); - - /* - int pos = 0; - while ((pos = copy.First (c)) != -1) - copy.RemoveAt (pos--); - */ + { + int pos = 0; + while ((pos = copy.firstIndexOf (String (c))) != -1) + copy.removeAt (pos--); + } return copy; } @@ -71,11 +67,13 @@ // String String::toUppercase() const { - String newstr = m_string; + String newstr (_string); for (char& c : newstr) + { if (c >= 'a' && c <= 'z') c -= 'a' - 'A'; + } return newstr; } @@ -84,11 +82,13 @@ // String String::toLowercase() const { - String newstr = m_string; + String newstr (_string); - for (char & c : newstr) + for (char& c : newstr) + { if (c >= 'A' && c <= 'Z') c += 'a' - 'A'; + } return newstr; } @@ -107,20 +107,16 @@ StringList String::split (const String& del) const { StringList res; - long a = 0; + int a = 0; + int b; // Find all separators and store the text left to them. - for (;;) + while ((b = firstIndexOf (del, a)) != -1) { - long b = firstIndexOf (del, a); - - if (b == -1) - break; - String sub = mid (a, b); if (sub.length() > 0) - res.append (mid (a, b)); + res << sub; a = b + del.length(); } @@ -139,7 +135,7 @@ long pos; while ((pos = firstIndexOf (a)) != -1) - m_string = m_string.replace (pos, strlen (a), b); + _string = _string.replace (pos, strlen (a), b); } // ============================================================================= @@ -148,7 +144,7 @@ { int needles = 0; - for (const char & c : m_string) + for (const char & c : _string) if (c == needle) needles++; @@ -174,7 +170,7 @@ } char* newstr = new char[b - a + 1]; - strncpy (newstr, m_string.c_str() + a, b - a); + strncpy (newstr, _string.c_str() + a, b - a); newstr[b - a] = '\0'; String other (newstr); @@ -190,7 +186,7 @@ for (int i = 0; i < length(); ++i) { - if (m_string[i] != ' ') + if (_string[i] != ' ') continue; if (++count < n) @@ -206,11 +202,12 @@ // int String::firstIndexOf (const char* c, int a) const { - for (; a < length(); a++) - if (m_string[a] == c[0] && strncmp (m_string.c_str() + a, c, strlen (c)) == 0) - return a; + int pos = _string.find (c, a); - return -1; + if (pos == (int) std::string::npos) + return -1; + + return pos; } // ============================================================================= @@ -221,7 +218,7 @@ a = length() - 1; for (; a > 0; a--) - if (m_string[a] == c[0] && strncmp (m_string.c_str() + a, c, strlen (c)) == 0) + if (_string[a] == c[0] && strncmp (_string.c_str() + a, c, strlen (c)) == 0) return a; return -1; @@ -231,10 +228,10 @@ // void String::dump() const { - print ("`%1`:\n", chars()); + print ("`%1`:\n", c_str()); int i = 0; - for (char u : m_string) + for (char u : _string) print ("\t%1. [%d2] `%3`\n", i++, u, String (u)); } @@ -244,7 +241,7 @@ { errno = 0; char* endptr; - long i = strtol (m_string.c_str(), &endptr, base); + long i = strtol (_string.c_str(), &endptr, base); if (ok) *ok = (errno == 0 && *endptr == '\0'); @@ -258,7 +255,7 @@ { errno = 0; char* endptr; - float i = strtof (m_string.c_str(), &endptr); + float i = strtof (_string.c_str(), &endptr); if (ok) *ok = (errno == 0 && *endptr == '\0'); @@ -272,7 +269,7 @@ { errno = 0; char* endptr; - double i = strtod (m_string.c_str(), &endptr); + double i = strtod (_string.c_str(), &endptr); if (ok) *ok = (errno == 0 && *endptr == '\0'); @@ -304,10 +301,10 @@ { bool gotDot = false; - for (const char & c : m_string) + for (const char & c : _string) { // Allow leading hyphen for negatives - if (&c == &m_string[0] && c == '-') + if (&c == &_string[0] && c == '-') continue; // Check for decimal point @@ -336,7 +333,7 @@ return false; const int ofs = length() - other.length(); - return strncmp (chars() + ofs, other.chars(), other.length()) == 0; + return strncmp (c_str() + ofs, other.c_str(), other.length()) == 0; } // ============================================================================= @@ -346,7 +343,7 @@ if (length() < other.length()) return false; - return strncmp (chars(), other.chars(), other.length()) == 0; + return strncmp (c_str(), other.c_str(), other.length()) == 0; } // ============================================================================= @@ -363,7 +360,7 @@ while (vsnprintf (buf, bufsize, fmtstr, va) >= bufsize); va_end (va); - m_string = buf; + _string = buf; delete[] buf; } @@ -391,10 +388,10 @@ // Elevate to uppercase for case-insensitive matching String pattern_upper = pattern.toUppercase(); String this_upper = toUppercase(); - const char* maskstring = pattern_upper.chars(); + const char* maskstring = pattern_upper.c_str(); const char* mptr = &maskstring[0]; - for (const char* sptr = this_upper.chars(); *sptr != '\0'; sptr++) + for (const char* sptr = this_upper.c_str(); *sptr != '\0'; sptr++) { if (*mptr == '?') {
--- a/src/string.h Fri May 02 21:42:11 2014 +0300 +++ b/src/string.h Sun May 04 18:31:40 2014 +0300 @@ -42,307 +42,89 @@ // class String { - public: - using StringType = std::string; - using Iterator = typename StringType::iterator; - using ConstIterator = StringType::const_iterator; - - String() {} - - explicit String (char a) : - m_string (&a) {} - - String (const char* data) : - m_string (data) {} - - String (const StringType& data) : - m_string (data) {} +public: + String() {} - void dump() const; - int compare (const String& other) const; - bool endsWith (const String& other); - int count (char needle) const; - int firstIndexOf (const char* c, int a = 0) const; - int lastIndexOf (const char* c, int a = -1) const; - String toLowercase() const; - bool isNumeric() const; - bool maskAgainst (const String& pattern) const; - int wordPosition (int n) const; - void replace (const char* a, const char* b); - StringList split (const String& del) const; - StringList split (char del) const; - void sprintf (const char* fmtstr, ...); - bool startsWith (const String& other); - String strip (const List<char>& unwanted); - String mid (long a, long b = -1) const; - double toDouble (bool* ok = nullptr) const; - float toFloat (bool* ok = nullptr) const; - long toLong (bool* ok = nullptr, int base = 10) const; - void trim (int n); - String toUppercase() const; - - String operator+ (const String& data) const; - String operator+ (const char* data) const; - - static String fromNumber (int a); - static String fromNumber (long a); - - inline bool isEmpty() const - { - return m_string[0] == '\0'; - } - - inline void append (const char* data) - { - m_string.append (data); - } - - inline void append (char data) - { - m_string.push_back (data); - } - - inline void append (const String& data) - { - m_string.append (data.chars()); - } - - inline Iterator begin() - { - return m_string.begin(); - } + explicit String (char a) : + _string ({ a, '\0' }) {} - inline ConstIterator begin() const - { - return m_string.cbegin(); - } - - inline const char* chars() const - { - return m_string.c_str(); - } - - inline const char* c_str() const - { - return m_string.c_str(); - } - - inline Iterator end() - { - return m_string.end(); - } - - inline ConstIterator end() const - { - return m_string.end(); - } - - inline void clear() - { - m_string.clear(); - } - - inline void removeAt (int pos) - { - m_string.erase (m_string.begin() + pos); - } + String (const char* data) : + _string (data) {} - inline void insert (int pos, char c) - { - m_string.insert (m_string.begin() + pos, c); - } - - inline int length() const - { - return m_string.length(); - } - - inline void remove (int pos, int len) - { - m_string.replace (pos, len, ""); - } - - inline void removeFromStart (int len) - { - remove (0, len); - } - - inline void removeFromEnd (int len) - { - remove (length() - len, len); - } - - inline void replace (int pos, int n, const String& a) - { - m_string.replace (pos, n, a.chars()); - } - - inline void shrinkToFit() - { - m_string.shrink_to_fit(); - } - - inline const StringType& stdString() const - { - return m_string; - } + String (const std::string& data) : + _string (data) {} - inline String strip (char unwanted) - { - return strip ({unwanted}); - } - - // ============================================================================= - // - inline String operator+ (int num) const - { - return *this + String::fromNumber (num); - } - - // ============================================================================= - // - inline String& operator+= (const String data) - { - append (data); - return *this; - } - - // ============================================================================= - // - inline String& operator+= (const char* data) - { - append (data); - return *this; - } - - // ============================================================================= - // - inline String& operator+= (int num) - { - return operator+= (String::fromNumber (num)); - } - - // ============================================================================= - // - inline void prepend (String a) - { - m_string = (a + m_string).stdString(); - } - - // ============================================================================= - // - inline String& operator+= (const char data) - { - append (data); - return *this; - } - - // ============================================================================= - // - inline String operator- (int n) const - { - String newString = m_string; - newString -= n; - return newString; - } - - // ============================================================================= - // - inline String& operator-= (int n) - { - trim (n); - return *this; - } - - // ============================================================================= - // - inline String operator+() const - { - return toUppercase(); - } + inline void append (const char* data); + inline void append (char data); + inline void append (const String& data); + inline std::string::iterator begin(); + inline std::string::const_iterator begin() const; + void dump() const; + inline void clear(); + int compare (const String& other) const; + int count (char needle) const; + inline const char* c_str() const; + inline std::string::iterator end(); + inline std::string::const_iterator end() const; + bool endsWith (const String& other); + int firstIndexOf (const char* c, int a = 0) const; + String toLowercase() const; + inline int indexDifference (int a, int b); + inline void insert (int pos, char c); + inline bool isEmpty() const; + bool isNumeric() const; + int lastIndexOf (const char* c, int a = -1) const; + inline int length() const; + bool maskAgainst (const String& pattern) const; + String mid (long a, long b = -1) const; + inline void modifyIndex (int& a); + inline void prepend (String a); + inline void removeAt (int pos); + inline void remove (int pos, int len); + inline void removeFromEnd (int len); + inline void removeFromStart (int len); + void replace (const char* a, const char* b); + inline void replace (int pos, int n, const String& a); + inline void shrinkToFit(); + StringList split (const String& del) const; + StringList split (char del) const; + void sprintf (const char* fmtstr, ...); + bool startsWith (const String& other); + inline const std::string& stdString() const; + inline String strip (char unwanted); + String strip (const List<char>& unwanted); + double toDouble (bool* ok = nullptr) const; + float toFloat (bool* ok = nullptr) const; + long toLong (bool* ok = nullptr, int base = 10) const; + void trim (int n); + String toUppercase() const; + int wordPosition (int n) const; - // ============================================================================= - // - inline String operator-() const - { - return toLowercase(); - } - - // ============================================================================= - // - inline bool operator== (const String& other) const - { - return stdString() == other.stdString(); - } - - // ============================================================================= - // - inline bool operator== (const char* other) const - { - return operator== (String (other)); - } - - // ============================================================================= - // - inline bool operator!= (const String& other) const - { - return stdString() != other.stdString(); - } - - // ============================================================================= - // - inline bool operator!= (const char* other) const - { - return operator!= (String (other)); - } + static String fromNumber (int a); + static String fromNumber (long a); - // ============================================================================= - // - inline bool operator> (const String& other) const - { - return stdString() > other.stdString(); - } - - // ============================================================================= - // - inline bool operator< (const String& other) const - { - return stdString() < other.stdString(); - } - - // ============================================================================= - // - inline operator const char*() const - { - return chars(); - } + String operator+ (const String& data) const; + String operator+ (const char* data) const; + inline String operator+ (int num) const; + inline String& operator+= (const String data); + inline String& operator+= (const char* data); + inline String& operator+= (int num); + inline String& operator+= (const char data); + inline String operator- (int n) const; + inline String& operator-= (int n); + inline bool operator== (const String& other) const; + inline bool operator== (const char* other) const; + inline bool operator!= (const String& other) const; + inline bool operator!= (const char* other) const; + inline bool operator> (const String& other) const; + inline bool operator< (const String& other) const; + inline operator const char*() const; + inline operator const std::string&() const; - // ============================================================================= - // - inline operator const StringType&() const - { - return stdString(); - } - - void modifyIndex (int& a) - { - if (a < 0) - a = length() - a; - } - - int indexDifference (int a, int b) - { - modifyIndex (a); - modifyIndex (b); - return b - a; - } - - private: - StringType m_string; +private: + std::string _string; }; -// ============================================================================= -// class StringList : public List<String> { public: @@ -355,17 +137,218 @@ String join (const String& delim); }; +inline bool operator== (const char* a, const String& b); +inline String operator+ (const char* a, const String& b); // ============================================================================= // +// IMPLEMENTATIONS +// + +inline bool String::isEmpty() const +{ + return _string[0] == '\0'; +} + +inline void String::append (const char* data) +{ + _string.append (data); +} + +inline void String::append (char data) +{ + _string.push_back (data); +} + +inline void String::append (const String& data) +{ + _string.append (data.c_str()); +} + +inline std::string::iterator String::begin() +{ + return _string.begin(); +} + +inline std::string::const_iterator String::begin() const +{ + return _string.cbegin(); +} + +inline const char* String::c_str() const +{ + return _string.c_str(); +} + +inline std::string::iterator String::end() +{ + return _string.end(); +} + +inline std::string::const_iterator String::end() const +{ + return _string.end(); +} + +inline void String::clear() +{ + _string.clear(); +} + +inline void String::removeAt (int pos) +{ + _string.erase (_string.begin() + pos); +} + +inline void String::insert (int pos, char c) +{ + _string.insert (_string.begin() + pos, c); +} + +inline int String::length() const +{ + return _string.length(); +} + +inline void String::remove (int pos, int len) +{ + _string.replace (pos, len, ""); +} + +inline void String::removeFromStart (int len) +{ + remove (0, len); +} + +inline void String::removeFromEnd (int len) +{ + remove (length() - len, len); +} + +inline void String::replace (int pos, int n, const String& a) +{ + _string.replace (pos, n, a.c_str()); +} + +inline void String::shrinkToFit() +{ + _string.shrink_to_fit(); +} + +inline const std::string& String::stdString() const +{ + return _string; +} + +inline String String::strip (char unwanted) +{ + return strip ({unwanted}); +} + +inline String String::operator+ (int num) const +{ + return *this + String::fromNumber (num); +} + +inline String& String::operator+= (const String data) +{ + append (data); + return *this; +} + +inline String& String::operator+= (const char* data) +{ + append (data); + return *this; +} + +inline String& String::operator+= (int num) +{ + return operator+= (String::fromNumber (num)); +} + +inline void String::prepend (String a) +{ + _string = (a + _string).stdString(); +} + +inline String& String::operator+= (const char data) +{ + append (data); + return *this; +} + +inline String String::operator- (int n) const +{ + String newString = _string; + newString -= n; + return newString; +} + +inline String& String::operator-= (int n) +{ + trim (n); + return *this; +} + +inline bool String::operator== (const String& other) const +{ + return stdString() == other.stdString(); +} + +inline bool String::operator== (const char* other) const +{ + return operator== (String (other)); +} + +inline bool String::operator!= (const String& other) const +{ + return stdString() != other.stdString(); +} + +inline bool String::operator!= (const char* other) const +{ + return operator!= (String (other)); +} + +inline bool String::operator> (const String& other) const +{ + return stdString() > other.stdString(); +} + +inline bool String::operator< (const String& other) const +{ + return stdString() < other.stdString(); +} + +inline String::operator const char*() const +{ + return c_str(); +} + +inline String::operator const std::string&() const +{ + return stdString(); +} + +inline void String::modifyIndex (int& a) +{ + if (a < 0) + a = length() - a; +} + +inline int String::indexDifference (int a, int b) +{ + modifyIndex (a); + modifyIndex (b); + return b - a; +} + inline bool operator== (const char* a, const String& b) { return b == a; } - -// ============================================================================= -// inline String operator+ (const char* a, const String& b) { return String (a) + b;