/** * to_string.hpp - Provides compile-time integer-to-string conversion. * Written by Clyne Sullivan. * https://github.com/tcsullivan/constexpr-to-string */ #ifndef TCSULLIVAN_TO_STRING_HPP_ #define TCSULLIVAN_TO_STRING_HPP_ #include /** * @struct to_string_t * @brief Provides the ability to convert any integral to a string at compile-time. * @tparam N Number to convert * @tparam base Desired base, can be from 2 to 36 */ template, int> = 0, std::enable_if_t<(base > 1 && base < 37), int> = 0> struct to_string_t { // The lambda calculates what the string length of N will be, so that `buf` // fits to the number perfectly. char_type buf[([] { unsigned int len = N > 0 ? 1 : 2; for (auto n = N < 0 ? -N : N; n; len++, n /= base); return len; }())] = {}; /** * Constructs the object, filling `buf` with the string representation of N. */ constexpr to_string_t() noexcept { if (N != 0) { auto ptr = buf + sizeof(buf) / sizeof(buf[0]); *--ptr = '\0'; for (auto n = N < 0 ? -N : N; n; n /= base) *--ptr = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ"[n % base]; if (N < 0) *--ptr = '-'; } else { buf[0] = '0'; } } // Support implicit casting to `char *` or `const char *`. constexpr operator char_type *() { return buf; } constexpr operator const char_type *() const { return buf; } // Support range-based for loops constexpr auto begin() { return buf; } constexpr auto begin() const { return buf; } constexpr auto end() { return buf + sizeof(buf) / sizeof(buf[0]); } constexpr auto end() const { return buf + sizeof(buf) / sizeof(buf[0]); } }; /** * Simplifies use of `to_string_t` from `to_string_t()` to `to_string`. */ template constexpr to_string_t to_string; #endif // TCSULLIVAN_TO_STRING_HPP_