#ifndef MARISA_STRING_H_ #define MARISA_STRING_H_ #include "base.h" namespace marisa { class String { public: String() : ptr_(NULL), length_(0) {} explicit String(const char *str) : ptr_(str), length_(0) { while (str[length_] != '\0') { ++length_; } } String(const char *ptr, std::size_t length) : ptr_(ptr), length_(length) {} String(const String &str) : ptr_(str.ptr_), length_(str.length_) {} String substr(std::size_t pos, std::size_t length) const { MARISA_DEBUG_IF(pos + length > length_, MARISA_PARAM_ERROR); return String(ptr_ + pos, length); } String &operator=(const String &str) { ptr_ = str.ptr_; length_ = str.length_; return *this; } UInt8 operator[](std::size_t i) const { MARISA_DEBUG_IF(i >= length_, MARISA_PARAM_ERROR); return ptr_[i]; } const char *ptr() const { return ptr_; } std::size_t length() const { return length_; } private: const char *ptr_; std::size_t length_; }; inline bool operator==(const String &lhs, const String &rhs) { if (lhs.length() != rhs.length()) { return false; } for (std::size_t i = 0; i < lhs.length(); ++i) { if (lhs[i] != rhs[i]) { return false; } } return true; } inline bool operator!=(const String &lhs, const String &rhs) { return !(lhs == rhs); } inline bool operator<(const String &lhs, const String &rhs) { for (std::size_t i = 0; i < lhs.length(); ++i) { if (i == rhs.length()) { return false; } if (lhs[i] != rhs[i]) { return lhs[i] < rhs[i]; } } return lhs.length() < rhs.length(); } inline bool operator>(const String &lhs, const String &rhs) { return rhs < lhs; } class RString { public: RString() : ptr_(static_cast<const char *>(NULL) - 1), length_(0) {} explicit RString(const String &str) : ptr_(str.ptr() + str.length() - 1), length_(str.length()) {} RString(const RString &str) : ptr_(str.ptr_), length_(str.length_) {} RString substr(std::size_t pos, std::size_t length) const { MARISA_DEBUG_IF(pos + length > length_, MARISA_PARAM_ERROR); RString str(*this); str.ptr_ -= pos; str.length_ = length; return str; } RString &operator=(const RString &str) { ptr_ = str.ptr_; length_ = str.length_; return *this; } UInt8 operator[](std::size_t i) const { MARISA_DEBUG_IF(i >= length_, MARISA_PARAM_ERROR); return *(ptr_ - i); } const char *ptr() const { return ptr_ - length_ + 1; } std::size_t length() const { return length_; } private: const char *ptr_; std::size_t length_; }; inline bool operator==(const RString &lhs, const RString &rhs) { if (lhs.length() != rhs.length()) { return false; } for (std::size_t i = 0; i < lhs.length(); ++i) { if (lhs[i] != rhs[i]) { return false; } } return true; } inline bool operator!=(const RString &lhs, const RString &rhs) { return !(lhs == rhs); } inline bool operator<(const RString &lhs, const RString &rhs) { for (std::size_t i = 0; i < lhs.length(); ++i) { if (i == rhs.length()) { return false; } if (lhs[i] != rhs[i]) { return lhs[i] < rhs[i]; } } return lhs.length() < rhs.length(); } inline bool operator>(const RString &lhs, const RString &rhs) { return rhs < lhs; } } // namespace marisa #endif // MARISA_STRING_H_