Zserio C++ runtime library  1.2.0
Built for Zserio 2.16.0
StringConvertUtil.h
Go to the documentation of this file.
1 #ifndef ZSERIO_STRING_CONVERT_UTIL_H_INC
2 #define ZSERIO_STRING_CONVERT_UTIL_H_INC
3 
4 #include <array>
5 #include <limits>
6 #include <sstream>
7 
8 #include "zserio/RebindAlloc.h"
9 #include "zserio/String.h"
10 
11 namespace zserio
12 {
13 
14 namespace detail
15 {
16 
24 template <typename T,
25  typename std::enable_if<std::is_unsigned<T>::value && !std::is_same<T, bool>::value, int>::type = 0>
26 const char* convertIntToString(std::array<char, 24>& buffer, T value, bool isNegative)
27 {
28  static const std::array<char, 201> DIGITS_100_10 = {
29  "0001020304050607080910111213141516171819"
30  "2021222324252627282930313233343536373839"
31  "4041424344454647484950515253545556575859"
32  "6061626364656667686970717273747576777879"
33  "8081828384858687888990919293949596979899"};
34  static const std::array<char, 11> DIGITS_1 = {"0123456789"};
35 
36  auto bufferEnd = buffer.end();
37  *(--bufferEnd) = '\0'; // always terminate with '\0'
38 
39  while (value >= 100)
40  {
41  const unsigned int index = static_cast<unsigned int>((value % 100) * 2);
42  value /= 100;
43  *(--bufferEnd) = DIGITS_100_10[index + 1];
44  *(--bufferEnd) = DIGITS_100_10[index];
45  }
46 
47  if (value < 10)
48  {
49  *(--bufferEnd) = DIGITS_1[static_cast<unsigned int>(value)];
50  }
51  else
52  {
53  const unsigned int index = static_cast<unsigned int>(value * 2);
54  *(--bufferEnd) = DIGITS_100_10[index + 1];
55  *(--bufferEnd) = DIGITS_100_10[index];
56  }
57 
58  if (isNegative)
59  {
60  *(--bufferEnd) = '-';
61  }
62 
63  return &(*bufferEnd);
64 }
65 
66 } // namespace detail
67 
78 template <typename T, typename std::enable_if<std::is_unsigned<T>::value, int>::type = 0>
79 const char* convertIntToString(std::array<char, 24>& buffer, T value)
80 {
81  return detail::convertIntToString(buffer, value, false);
82 }
83 
94 template <typename T, typename std::enable_if<std::is_signed<T>::value, int>::type = 0>
95 const char* convertIntToString(std::array<char, 24>& buffer, T value)
96 {
97  using unsigned_type = typename std::make_unsigned<T>::type;
98  unsigned_type absValue = static_cast<unsigned_type>(value);
99  const bool isNegative = value < 0;
100  if (isNegative)
101  {
102  absValue = static_cast<unsigned_type>(0 - absValue);
103  }
104 
105  return detail::convertIntToString(buffer, absValue, isNegative);
106 }
107 
119 inline void convertFloatToString(std::array<char, 24>& integerPartBuffer,
120  std::array<char, 24>& floatingPartBuffer, float value, const char*& integerPartString,
121  const char*& floatingPartString)
122 {
123  if (value >= static_cast<float>(std::numeric_limits<int64_t>::max()))
124  {
125  integerPartString = "+Inf";
126  floatingPartString = nullptr;
127  }
128  else if (value <= static_cast<float>(std::numeric_limits<int64_t>::min()))
129  {
130  integerPartString = "-Inf";
131  floatingPartString = nullptr;
132  }
133  else
134  {
135  const int64_t integerPart = static_cast<int64_t>(value);
136  const int64_t floatingPart =
137  static_cast<int64_t>((value - static_cast<float>(integerPart)) * 1e3F); // 3 digits
138  const int64_t floatingPartAbs = (floatingPart < 0) ? 0 - floatingPart : floatingPart;
139  integerPartString = convertIntToString(integerPartBuffer, integerPart);
140  floatingPartString = convertIntToString(floatingPartBuffer, floatingPartAbs);
141  }
142 }
143 
151 inline const char* convertBoolToString(bool value)
152 {
153  return value ? "true" : "false";
154 }
155 
164 template <typename ALLOC, typename T>
165 string<ALLOC> toString(T value, const ALLOC& allocator = ALLOC())
166 {
167  std::array<char, 24> buffer = {};
168  return string<ALLOC>(convertIntToString(buffer, value), allocator);
169 }
170 
180 template <typename ALLOC>
181 string<ALLOC> toString(bool value, const ALLOC& allocator = ALLOC())
182 {
183  return string<ALLOC>(convertBoolToString(value), allocator);
184 }
185 
193 template <typename T>
195 {
196  return toString<std::allocator<char>>(value);
197 }
198 
199 } // namespace zserio
200 
201 #endif // ifndef ZSERIO_STRING_CONVERT_UTIL_H_INC
const char * convertIntToString(std::array< char, 24 > &buffer, T value)
std::basic_string< char, std::char_traits< char >, RebindAlloc< ALLOC, char > > string
Definition: String.h:17
const char * convertBoolToString(bool value)
void convertFloatToString(std::array< char, 24 > &integerPartBuffer, std::array< char, 24 > &floatingPartBuffer, float value, const char *&integerPartString, const char *&floatingPartString)
string< ALLOC > toString(T value, const ALLOC &allocator=ALLOC())