feat(string): add parsing of arithmetic types from strings
This commit is contained in:
parent
211a30525e
commit
d8803880f1
@ -6,6 +6,8 @@
|
|||||||
#include "String/StringView.h"
|
#include "String/StringView.h"
|
||||||
#include "Miscellaneous/AssertionMacros.h"
|
#include "Miscellaneous/AssertionMacros.h"
|
||||||
|
|
||||||
|
#include <cmath>
|
||||||
|
|
||||||
NAMESPACE_REDCRAFT_BEGIN
|
NAMESPACE_REDCRAFT_BEGIN
|
||||||
NAMESPACE_MODULE_BEGIN(Redcraft)
|
NAMESPACE_MODULE_BEGIN(Redcraft)
|
||||||
NAMESPACE_MODULE_BEGIN(Utility)
|
NAMESPACE_MODULE_BEGIN(Utility)
|
||||||
@ -17,6 +19,7 @@ void TestString()
|
|||||||
TestChar();
|
TestChar();
|
||||||
TestStringView();
|
TestStringView();
|
||||||
TestTemplateString();
|
TestTemplateString();
|
||||||
|
TestStringConversion();
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestChar()
|
void TestChar()
|
||||||
@ -553,6 +556,110 @@ void TestTemplateString()
|
|||||||
Test(InPlaceType<unicodechar>);
|
Test(InPlaceType<unicodechar>);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestStringConversion()
|
||||||
|
{
|
||||||
|
auto Test = []<typename T>(TInPlaceType<T>)
|
||||||
|
{
|
||||||
|
auto CheckParseArithmetic = []<typename U>(TStringView<T> View, U Result)
|
||||||
|
{
|
||||||
|
U Object;
|
||||||
|
|
||||||
|
always_check(View.Parse(LITERAL(T, "{0:}"), Object) == 1);
|
||||||
|
|
||||||
|
if constexpr (CFloatingPoint<U>)
|
||||||
|
{
|
||||||
|
always_check(NAMESPACE_STD::isinf(Result) == NAMESPACE_STD::isinf(Object));
|
||||||
|
always_check(NAMESPACE_STD::isnan(Result) == NAMESPACE_STD::isnan(Object));
|
||||||
|
|
||||||
|
always_check(NAMESPACE_STD::signbit(Result) == NAMESPACE_STD::signbit(Object));
|
||||||
|
|
||||||
|
if (NAMESPACE_STD::isinf(Result) || NAMESPACE_STD::isnan(Result)) return;
|
||||||
|
|
||||||
|
constexpr auto Epsilon = 1e-3;
|
||||||
|
|
||||||
|
always_check(NAMESPACE_STD::abs(Object - Result) < Epsilon);
|
||||||
|
}
|
||||||
|
else always_check(Object == Result);
|
||||||
|
};
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "true" ), true );
|
||||||
|
CheckParseArithmetic(LITERAL(T, "false"), false);
|
||||||
|
|
||||||
|
auto CheckParseInt = [&]<typename U>(TInPlaceType<U>)
|
||||||
|
{
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+0"), static_cast<U>(+0.0));
|
||||||
|
CheckParseArithmetic(LITERAL(T, " 0"), static_cast<U>( 0.0));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-0"), static_cast<U>(-0.0));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+42"), static_cast<U>( +42));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+052"), static_cast<U>( +052));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+0x2A"), static_cast<U>( +0x2A));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+0b101010"), static_cast<U>(+0b101010));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "42"), static_cast<U>( 42));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "052"), static_cast<U>( 052));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "0x2A"), static_cast<U>( 0x2A));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "0b101010"), static_cast<U>(0b101010));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-42"), static_cast<U>( -42));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-052"), static_cast<U>( -052));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-0x2A"), static_cast<U>( -0x2A));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-0b101010"), static_cast<U>(-0b101010));
|
||||||
|
};
|
||||||
|
|
||||||
|
CheckParseInt(InPlaceType<bool>);
|
||||||
|
|
||||||
|
CheckParseInt(InPlaceType<int8>);
|
||||||
|
CheckParseInt(InPlaceType<int16>);
|
||||||
|
CheckParseInt(InPlaceType<int32>);
|
||||||
|
CheckParseInt(InPlaceType<int64>);
|
||||||
|
|
||||||
|
CheckParseInt(InPlaceType<uint8>);
|
||||||
|
CheckParseInt(InPlaceType<uint16>);
|
||||||
|
CheckParseInt(InPlaceType<uint32>);
|
||||||
|
CheckParseInt(InPlaceType<uint64>);
|
||||||
|
|
||||||
|
auto CheckParseFloat = [&]<typename U>(TInPlaceType<U>)
|
||||||
|
{
|
||||||
|
CheckParseInt(InPlaceType<U>);
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+3.14"), static_cast<U>( +3.14));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+3.14e2"), static_cast<U>( +3.14e2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+3.14e-2"), static_cast<U>( +3.14e-2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+0x1.91eb86p1"), static_cast<U>(+0x1.91eb86p1));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "3.14"), static_cast<U>( 3.14));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "3.14e2"), static_cast<U>( 3.14e2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "3.14e-2"), static_cast<U>( 3.14e-2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "0x1.91eb86p1"), static_cast<U>(0x1.91eb86p1));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-3.14"), static_cast<U>( -3.14));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-3.14e2"), static_cast<U>( -3.14e2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-3.14e-2"), static_cast<U>( -3.14e-2));
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-0x1.91eb86p1"), static_cast<U>(-0x1.91eb86p1));
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+Infinity"), +NAMESPACE_STD::numeric_limits<U>::infinity());
|
||||||
|
CheckParseArithmetic(LITERAL(T, " Infinity"), +NAMESPACE_STD::numeric_limits<U>::infinity());
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-Infinity"), -NAMESPACE_STD::numeric_limits<U>::infinity());
|
||||||
|
|
||||||
|
CheckParseArithmetic(LITERAL(T, "+NaN"), +NAMESPACE_STD::numeric_limits<U>::quiet_NaN());
|
||||||
|
CheckParseArithmetic(LITERAL(T, " NaN"), +NAMESPACE_STD::numeric_limits<U>::quiet_NaN());
|
||||||
|
CheckParseArithmetic(LITERAL(T, "-NaN"), -NAMESPACE_STD::numeric_limits<U>::quiet_NaN());
|
||||||
|
};
|
||||||
|
|
||||||
|
CheckParseFloat(InPlaceType<float>);
|
||||||
|
CheckParseFloat(InPlaceType<double>);
|
||||||
|
CheckParseFloat(InPlaceType<long double>);
|
||||||
|
};
|
||||||
|
|
||||||
|
Test(InPlaceType<char>);
|
||||||
|
Test(InPlaceType<wchar>);
|
||||||
|
Test(InPlaceType<u8char>);
|
||||||
|
Test(InPlaceType<u16char>);
|
||||||
|
Test(InPlaceType<u32char>);
|
||||||
|
Test(InPlaceType<unicodechar>);
|
||||||
|
}
|
||||||
|
|
||||||
NAMESPACE_END(Testing)
|
NAMESPACE_END(Testing)
|
||||||
|
|
||||||
NAMESPACE_MODULE_END(Utility)
|
NAMESPACE_MODULE_END(Utility)
|
||||||
|
@ -58,6 +58,197 @@ struct TStringHelper
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
else if constexpr (CArithmetic<U>)
|
||||||
|
{
|
||||||
|
checkf(Fmt.IsEmpty(), TEXT("Formatted parsing of arithmetic types not implemented."));
|
||||||
|
|
||||||
|
// Skip leading white spaces.
|
||||||
|
while (!View.IsEmpty() && TChar<T>::IsSpace(View.Front())) View.RemovePrefix(1);
|
||||||
|
|
||||||
|
if (View.IsEmpty()) return false;
|
||||||
|
|
||||||
|
bool bNegative = false;
|
||||||
|
|
||||||
|
// Handle optional sign.
|
||||||
|
if (View.Front() == LITERAL(T, '+'))
|
||||||
|
{
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
else if (View.Front() == LITERAL(T, '-'))
|
||||||
|
{
|
||||||
|
bNegative = true;
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle boolean conversion.
|
||||||
|
else if constexpr (CSameAs<U, bool>)
|
||||||
|
{
|
||||||
|
bool bIsTrue = false;
|
||||||
|
bool bIsFalse = false;
|
||||||
|
|
||||||
|
bIsTrue |= View.StartsWith(LITERAL(T, "true")) || View.StartsWith(LITERAL(T, "True")) || View.StartsWith(LITERAL(T, "TRUE"));
|
||||||
|
bIsFalse |= View.StartsWith(LITERAL(T, "false")) || View.StartsWith(LITERAL(T, "False")) || View.StartsWith(LITERAL(T, "FALSE"));
|
||||||
|
|
||||||
|
if (bIsTrue) { View.RemovePrefix(4); Object = true; return true; }
|
||||||
|
if (bIsFalse) { View.RemovePrefix(5); Object = false; return true; }
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle floating-point conversion.
|
||||||
|
if constexpr (CFloatingPoint<U>)
|
||||||
|
{
|
||||||
|
bool bIsInfinity = false;
|
||||||
|
bool bIsNaN = false;
|
||||||
|
|
||||||
|
bIsInfinity |= View.StartsWith(LITERAL(T, "infinity")) || View.StartsWith(LITERAL(T, "Infinity")) || View.StartsWith(LITERAL(T, "INFINITY"));
|
||||||
|
bIsNaN |= View.StartsWith(LITERAL(T, "nan")) || View.StartsWith(LITERAL(T, "NaN")) || View.StartsWith(LITERAL(T, "NAN"));
|
||||||
|
|
||||||
|
if (bIsInfinity) { View.RemovePrefix(8); Object = bNegative ? -NAMESPACE_STD::numeric_limits<U>::infinity() : NAMESPACE_STD::numeric_limits<U>::infinity(); return true; }
|
||||||
|
if (bIsNaN) { View.RemovePrefix(3); Object = bNegative ? -NAMESPACE_STD::numeric_limits<U>::quiet_NaN() : NAMESPACE_STD::numeric_limits<U>::quiet_NaN(); return true; }
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned Base = 0;
|
||||||
|
|
||||||
|
// Auto detect base.
|
||||||
|
{
|
||||||
|
if (View.Num() >= 2 && View.Front() == LITERAL(T, '0'))
|
||||||
|
{
|
||||||
|
if (View[1] == LITERAL(T, 'x') || View[1] == LITERAL(T, 'X'))
|
||||||
|
{
|
||||||
|
Base = 16;
|
||||||
|
View.RemovePrefix(2);
|
||||||
|
}
|
||||||
|
else if (View[1] == LITERAL(T, 'b') || View[1] == LITERAL(T, 'B'))
|
||||||
|
{
|
||||||
|
Base = 2;
|
||||||
|
View.RemovePrefix(2);
|
||||||
|
}
|
||||||
|
else if (TChar<T>::IsDigit(View.Front(), 8))
|
||||||
|
{
|
||||||
|
Base = 8;
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
else Base = 10;
|
||||||
|
}
|
||||||
|
else Base = 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the number.
|
||||||
|
auto ToNumber = [&View]<typename NumberType>(TInPlaceType<NumberType>, unsigned Base, NumberType Init = static_cast<NumberType>(0)) -> NumberType
|
||||||
|
{
|
||||||
|
NumberType Result = Init;
|
||||||
|
|
||||||
|
while (!View.IsEmpty() && (Base == 10 ? TChar<T>::IsDigit(View.Front()) : TChar<T>::IsDigit(View.Front(), Base)))
|
||||||
|
{
|
||||||
|
Result = static_cast<NumberType>(Result * Base + *TChar<T>::ToDigit(View.Front()));
|
||||||
|
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Result;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle integral conversion.
|
||||||
|
if constexpr (CIntegral<U>)
|
||||||
|
{
|
||||||
|
using UnsignedU = TMakeUnsigned<TConditional<!CSameAs<U, bool>, U, int>>;
|
||||||
|
|
||||||
|
if (View.IsEmpty()) return false;
|
||||||
|
|
||||||
|
// The integral number must start with a digit.
|
||||||
|
if (!TChar<T>::IsDigit(View.Front(), Base)) return false;
|
||||||
|
|
||||||
|
// Parse the integral number.
|
||||||
|
UnsignedU Number = ToNumber(InPlaceType<UnsignedU>, Base);
|
||||||
|
|
||||||
|
Object = static_cast<U>(bNegative ? -Number : Number);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle floating-point conversion.
|
||||||
|
else if constexpr (CFloatingPoint<U>)
|
||||||
|
{
|
||||||
|
if (View.IsEmpty()) return false;
|
||||||
|
|
||||||
|
// The floating-point number must start with a digit or a dot.
|
||||||
|
if (!(TChar<T>::IsDigit(View.Front(), Base) || View.Front() == LITERAL(T, '.'))) return false;
|
||||||
|
|
||||||
|
size_t IntegralBeginNum = View.Num();
|
||||||
|
|
||||||
|
// Parse the integral number.
|
||||||
|
Object = ToNumber(InPlaceType<U>, Base);
|
||||||
|
|
||||||
|
size_t IntegralLength = IntegralBeginNum - View.Num();
|
||||||
|
|
||||||
|
// Parse the fractional number.
|
||||||
|
if (!View.IsEmpty() && View.Front() == LITERAL(T, '.'))
|
||||||
|
{
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
|
||||||
|
U InvBase = 1 / static_cast<U>(Base);
|
||||||
|
|
||||||
|
size_t FractionBeginNum = View.Num();
|
||||||
|
|
||||||
|
Object = ToNumber(InPlaceType<U>, Base, Object);
|
||||||
|
|
||||||
|
size_t FractionLength = FractionBeginNum - View.Num();
|
||||||
|
|
||||||
|
Object *= NAMESPACE_STD::pow(InvBase, static_cast<U>(FractionLength));
|
||||||
|
}
|
||||||
|
else if (IntegralLength == 0) return false;
|
||||||
|
|
||||||
|
// For floating point numbers apply the symbols directly
|
||||||
|
Object = static_cast<U>(bNegative ? -Object : Object);
|
||||||
|
|
||||||
|
if (View.IsEmpty()) return true;
|
||||||
|
|
||||||
|
if (Base != 10 && Base != 16) return true;
|
||||||
|
|
||||||
|
bool bHasExponent = false;
|
||||||
|
|
||||||
|
bHasExponent |= Base == 10 && View.Front() == LITERAL(T, 'e');
|
||||||
|
bHasExponent |= Base == 10 && View.Front() == LITERAL(T, 'E');
|
||||||
|
bHasExponent |= Base == 16 && View.Front() == LITERAL(T, 'p');
|
||||||
|
bHasExponent |= Base == 16 && View.Front() == LITERAL(T, 'P');
|
||||||
|
|
||||||
|
if (!bHasExponent) return true;
|
||||||
|
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
|
||||||
|
if (View.IsEmpty()) return false;
|
||||||
|
|
||||||
|
// Parse the exponent number.
|
||||||
|
{
|
||||||
|
bool bNegativeExponent = false;
|
||||||
|
|
||||||
|
if (View.Front() == LITERAL(T, '+'))
|
||||||
|
{
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
else if (View.Front() == LITERAL(T, '-'))
|
||||||
|
{
|
||||||
|
bNegativeExponent = true;
|
||||||
|
View.RemovePrefix(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// The exponent number must start with a digit.
|
||||||
|
if (!TChar<T>::IsDigit(View.Front())) return false;
|
||||||
|
|
||||||
|
U Exponent = ToNumber(InPlaceType<U>, 10);
|
||||||
|
|
||||||
|
Exponent = bNegativeExponent ? -Exponent : Exponent;
|
||||||
|
|
||||||
|
Object *= static_cast<U>(NAMESPACE_STD::pow(static_cast<U>(Base == 16 ? 2 : 10), Exponent));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
else static_assert(sizeof(U) == -1, "Unsupported arithmetic type");
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@ REDCRAFTUTILITY_API void TestString();
|
|||||||
REDCRAFTUTILITY_API void TestChar();
|
REDCRAFTUTILITY_API void TestChar();
|
||||||
REDCRAFTUTILITY_API void TestStringView();
|
REDCRAFTUTILITY_API void TestStringView();
|
||||||
REDCRAFTUTILITY_API void TestTemplateString();
|
REDCRAFTUTILITY_API void TestTemplateString();
|
||||||
|
REDCRAFTUTILITY_API void TestStringConversion();
|
||||||
|
|
||||||
NAMESPACE_END(Testing)
|
NAMESPACE_END(Testing)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user