Arbutils/src/ConstString.hpp

80 lines
3.6 KiB
C++
Raw Normal View History

2020-02-26 11:57:18 +00:00
#ifndef ARBUTILS_CONSTSTRING_HPP
#define ARBUTILS_CONSTSTRING_HPP
#include <cstddef>
#include <cstdint>
#include <cstring>
#include <string>
2020-02-26 11:57:18 +00:00
namespace Arbutils {
class ConstString {
private:
const char* _str;
size_t _length;
uint32_t _hash;
inline static uint32_t constexpr Hash(char const* input) {
return (*input) ? static_cast<uint32_t>((*input)) + 33 * Hash(input + 1) : 5381;
}
inline static constexpr char charToLower(const char c) { return (c >= 'A' && c <= 'Z') ? c + ('a' - 'A') : c; }
inline static uint32_t constexpr LowercaseHash(char const* input) {
return charToLower(*input) ? static_cast<uint32_t>(charToLower(*input)) + 33 * LowercaseHash(input + 1)
: 5381;
}
2020-02-26 11:57:18 +00:00
inline static int constexpr Length(const char* str) { return *str ? 1 + Length(str + 1) : 0; }
constexpr explicit ConstString(const char* str, bool)
: _str(str), _length(Length(str)), _hash(LowercaseHash(str)){};
constexpr explicit ConstString(const char* str, size_t size, bool)
: _str(str), _length(size), _hash(LowercaseHash(str)){};
2020-02-26 11:57:18 +00:00
public:
constexpr ConstString() : _str(""), _length(0), _hash(Hash("")){};
2020-02-26 11:57:18 +00:00
constexpr explicit ConstString(const char* str) : _str(str), _length(Length(str)), _hash(Hash(str)){};
constexpr explicit ConstString(const char* str, size_t size) : _str(str), _length(size), _hash(Hash(str)){};
2020-02-26 11:57:18 +00:00
static constexpr inline ConstString CaseInsensitive(const char* str) { return ConstString(str, true); }
static constexpr inline ConstString CaseInsensitive(const char* str, size_t size) {
return ConstString(str, size, true);
}
[[nodiscard]] inline constexpr const char* c_str() const noexcept { return _str; }
[[nodiscard]] inline std::string std_str() const { return std::string(_str, _length); }
2020-02-26 11:57:18 +00:00
[[nodiscard]] inline constexpr size_t Length() const noexcept { return _length; }
2020-02-26 11:57:18 +00:00
[[nodiscard]] inline constexpr uint32_t GetHash() const noexcept { return _hash; }
constexpr std::size_t operator()(ConstString const& s) const noexcept { return s.GetHash(); }
inline constexpr operator uint32_t() const { return _hash; }
inline constexpr bool operator==(const ConstString& rhs) const { return _hash == rhs._hash; }
inline constexpr bool operator!=(const ConstString& rhs) const { return _hash != rhs._hash; }
inline constexpr bool operator==(const std::string& rhs) const { return _hash == Hash(rhs.c_str()); }
inline constexpr bool operator!=(const std::string& rhs) const { return _hash != Hash(rhs.c_str()); }
inline constexpr bool operator==(const char* rhs) const { return _hash == Hash(rhs); }
inline constexpr bool operator!=(const char* rhs) const { return _hash != Hash(rhs); }
};
}
inline constexpr Arbutils::ConstString operator"" _const(const char* c, size_t l) {
return Arbutils::ConstString(c, l);
}
inline constexpr Arbutils::ConstString operator"" _c(const char* c, size_t l) { return Arbutils::ConstString(c, l); }
inline constexpr Arbutils::ConstString operator"" _const_nocase(const char* c, size_t l) {
return Arbutils::ConstString::CaseInsensitive(c, l);
}
inline constexpr Arbutils::ConstString operator"" _cnc(const char* c, size_t l) {
return Arbutils::ConstString::CaseInsensitive(c, l);
}
namespace std {
template <> struct hash<Arbutils::ConstString> {
constexpr std::size_t operator()(Arbutils::ConstString const& s) const noexcept { return s.GetHash(); }
2020-02-26 11:57:18 +00:00
};
}
#endif // ARBUTILS_CONSTSTRING_HPP