refactor(string): add base-checking to conversions between characters and digits
This commit is contained in:
parent
09bbcecc28
commit
f8ef1da107
@ -77,13 +77,13 @@ void TestChar()
|
|||||||
always_check(TChar<T>::ToLower(LITERAL(T, 'i')) == LITERAL(T, 'i'));
|
always_check(TChar<T>::ToLower(LITERAL(T, 'i')) == LITERAL(T, 'i'));
|
||||||
always_check(TChar<T>::ToUpper(LITERAL(T, 'l')) == LITERAL(T, 'L'));
|
always_check(TChar<T>::ToUpper(LITERAL(T, 'l')) == LITERAL(T, 'L'));
|
||||||
|
|
||||||
always_check(0x0 == TChar<T>::ToDigit(LITERAL(T, '0')));
|
always_check(0x0 == TChar<T>::ToDigit(LITERAL(T, '0'), 16));
|
||||||
always_check(0xF == TChar<T>::ToDigit(LITERAL(T, 'f')));
|
always_check(0xF == TChar<T>::ToDigit(LITERAL(T, 'f'), 16));
|
||||||
always_check(0xF == TChar<T>::ToDigit(LITERAL(T, 'F')));
|
always_check(0xF == TChar<T>::ToDigit(LITERAL(T, 'F'), 16));
|
||||||
|
|
||||||
always_check(LITERAL(T, '0') == TChar<T>::FromDigit(0x0));
|
always_check(LITERAL(T, '0') == TChar<T>::FromDigit(0x0, 16));
|
||||||
always_check(LITERAL(T, 'f') != TChar<T>::FromDigit(0xF));
|
always_check(LITERAL(T, 'f') != TChar<T>::FromDigit(0xF, 16));
|
||||||
always_check(LITERAL(T, 'F') == TChar<T>::FromDigit(0xF));
|
always_check(LITERAL(T, 'F') == TChar<T>::FromDigit(0xF, 16));
|
||||||
};
|
};
|
||||||
|
|
||||||
Test(InPlaceType<char>);
|
Test(InPlaceType<char>);
|
||||||
|
@ -798,79 +798,42 @@ struct TChar
|
|||||||
return InChar;
|
return InChar;
|
||||||
}
|
}
|
||||||
|
|
||||||
NODISCARD FORCEINLINE static constexpr TOptional<unsigned> ToDigit(CharType InChar)
|
NODISCARD FORCEINLINE static constexpr TOptional<unsigned> ToDigit(CharType InChar, unsigned Base = 10)
|
||||||
{
|
{
|
||||||
switch (InChar)
|
constexpr uint8 DigitFromChar[] =
|
||||||
{
|
{
|
||||||
case LITERAL(CharType, '0'): return 0;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '1'): return 1;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '2'): return 2;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '3'): return 3;
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '4'): return 4;
|
0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18,
|
||||||
case LITERAL(CharType, '5'): return 5;
|
0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '6'): return 6;
|
0xFF, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18,
|
||||||
case LITERAL(CharType, '7'): return 7;
|
0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '8'): return 8;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, '9'): return 9;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'a'): return 10;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'b'): return 11;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'c'): return 12;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'd'): return 13;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'e'): return 14;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'f'): return 15;
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
case LITERAL(CharType, 'g'): return 16;
|
};
|
||||||
case LITERAL(CharType, 'h'): return 17;
|
|
||||||
case LITERAL(CharType, 'i'): return 18;
|
static_assert(sizeof(DigitFromChar) == 256);
|
||||||
case LITERAL(CharType, 'j'): return 19;
|
|
||||||
case LITERAL(CharType, 'k'): return 20;
|
if constexpr (sizeof(CharType) > 1) if (InChar >> 8) return Invalid;
|
||||||
case LITERAL(CharType, 'l'): return 21;
|
|
||||||
case LITERAL(CharType, 'm'): return 22;
|
if (DigitFromChar[InChar] >= Base) return Invalid;
|
||||||
case LITERAL(CharType, 'n'): return 23;
|
|
||||||
case LITERAL(CharType, 'o'): return 24;
|
check(TChar::IsASCII(InChar));
|
||||||
case LITERAL(CharType, 'p'): return 25;
|
|
||||||
case LITERAL(CharType, 'q'): return 26;
|
return DigitFromChar[InChar];
|
||||||
case LITERAL(CharType, 'r'): return 27;
|
|
||||||
case LITERAL(CharType, 's'): return 28;
|
|
||||||
case LITERAL(CharType, 't'): return 29;
|
|
||||||
case LITERAL(CharType, 'u'): return 30;
|
|
||||||
case LITERAL(CharType, 'v'): return 31;
|
|
||||||
case LITERAL(CharType, 'w'): return 32;
|
|
||||||
case LITERAL(CharType, 'x'): return 33;
|
|
||||||
case LITERAL(CharType, 'y'): return 34;
|
|
||||||
case LITERAL(CharType, 'z'): return 35;
|
|
||||||
case LITERAL(CharType, 'A'): return 10;
|
|
||||||
case LITERAL(CharType, 'B'): return 11;
|
|
||||||
case LITERAL(CharType, 'C'): return 12;
|
|
||||||
case LITERAL(CharType, 'D'): return 13;
|
|
||||||
case LITERAL(CharType, 'E'): return 14;
|
|
||||||
case LITERAL(CharType, 'F'): return 15;
|
|
||||||
case LITERAL(CharType, 'G'): return 16;
|
|
||||||
case LITERAL(CharType, 'H'): return 17;
|
|
||||||
case LITERAL(CharType, 'I'): return 18;
|
|
||||||
case LITERAL(CharType, 'J'): return 19;
|
|
||||||
case LITERAL(CharType, 'K'): return 20;
|
|
||||||
case LITERAL(CharType, 'L'): return 21;
|
|
||||||
case LITERAL(CharType, 'M'): return 22;
|
|
||||||
case LITERAL(CharType, 'N'): return 23;
|
|
||||||
case LITERAL(CharType, 'O'): return 24;
|
|
||||||
case LITERAL(CharType, 'P'): return 25;
|
|
||||||
case LITERAL(CharType, 'Q'): return 26;
|
|
||||||
case LITERAL(CharType, 'R'): return 27;
|
|
||||||
case LITERAL(CharType, 'S'): return 28;
|
|
||||||
case LITERAL(CharType, 'T'): return 29;
|
|
||||||
case LITERAL(CharType, 'U'): return 30;
|
|
||||||
case LITERAL(CharType, 'V'): return 31;
|
|
||||||
case LITERAL(CharType, 'W'): return 32;
|
|
||||||
case LITERAL(CharType, 'X'): return 33;
|
|
||||||
case LITERAL(CharType, 'Y'): return 34;
|
|
||||||
case LITERAL(CharType, 'Z'): return 35;
|
|
||||||
default: return Invalid;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NODISCARD FORCEINLINE static constexpr TOptional<CharType> FromDigit(int InDigit)
|
NODISCARD FORCEINLINE static constexpr TOptional<CharType> FromDigit(unsigned InDigit, unsigned Base = 10)
|
||||||
{
|
{
|
||||||
if (InDigit < 0 || InDigit > 35) return Invalid;
|
if (InDigit > Base) return Invalid;
|
||||||
|
|
||||||
return LITERAL(CharType, "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ")[InDigit];
|
return LITERAL(CharType, "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ")[InDigit];
|
||||||
}
|
}
|
||||||
|
@ -209,9 +209,13 @@ struct TStringHelper
|
|||||||
{
|
{
|
||||||
NumberType Result = Init;
|
NumberType Result = Init;
|
||||||
|
|
||||||
while (!View.IsEmpty() && (Base == 10 ? TChar<T>::IsDigit(View.Front()) : TChar<T>::IsDigit(View.Front(), Base)))
|
while (!View.IsEmpty())
|
||||||
{
|
{
|
||||||
Result = static_cast<NumberType>(Result * Base + *TChar<T>::ToDigit(View.Front()));
|
auto Digit = TChar<T>::ToDigit(View.Front(), Base);
|
||||||
|
|
||||||
|
if (!Digit) break;
|
||||||
|
|
||||||
|
Result = Result * static_cast<NumberType>(Base) + static_cast<NumberType>(*Digit);
|
||||||
|
|
||||||
View.RemovePrefix(1);
|
View.RemovePrefix(1);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user