Как я могу расширить лексическое приведение для поддержки перечислимых типов?
У меня есть следующая функция, которая преобразует строку в числовой тип данных:
template <typename T>
bool ConvertString(const std::string& theString, T& theResult)
{
std::istringstream iss(theString);
return !(iss >> theResult).fail();
}
Однако это не работает для перечислимых типов, поэтому я сделал что-то вроде этого:
template <typename T>
bool ConvertStringToEnum(const std::string& theString, T& theResult)
{
std::istringstream iss(theString);
unsigned int temp;
const bool isValid = !(iss >> temp).fail();
theResult = static_cast<T>(temp);
return isValid;
}
(Я предполагаю, что theString имеет допустимое значение для перечислимого типа; я использую это в основном для простой сериализации)
Есть ли способ создать одну функцию, которая сочетает в себе оба из них?
Я немного поиграл с аргументами шаблона, но ничего не придумал; было бы неплохо не вызывать одну функцию для перечисляемых типов и другую для всего остального.
Спасибо
2 ответа
Вы должны сделать два шага. Нахождение целочисленного типа, достаточно большого для хранения значений. Вы могли бы использовать unsigned long
, но значения могут быть отрицательными. Тогда вы могли бы использовать long
но значения могут простираться в диапазон unsigned long
, Так что на самом деле нет подходящего типа.
Хотя есть хитрость с использованием разрешения перегрузки. Вот
template<typename T>
struct id { typedef T type; };
id<char[1]>::type &find_etype(int);
id<char[2]>::type &find_etype(unsigned int);
id<char[3]>::type &find_etype(long);
id<char[4]>::type &find_etype(unsigned long);
Вы можете изменить его соответствующим образом, чтобы покрыть также long long
или же unsigned long long
если ваша реализация имеет поддержку для этого. Теперь при передаче типа enum предпочтение будет отдаваться одному из них по сравнению со всеми остальными - это тип, который может хранить все его значения. Вам просто нужно пройти sizeof
типа возврата в некоторый шаблон.
template<int> struct get_etype;
template<> struct get_etype<1> { typedef int type; };
template<> struct get_etype<2> { typedef unsigned int type; };
template<> struct get_etype<3> { typedef long type; };
template<> struct get_etype<4> { typedef unsigned long type; };
Теперь вы можете получить правильный тип. Все, что вам нужно сейчас, это посмотреть, является ли какой-то тип перечислением. Как это сделать, описано в книге "Шаблоны C++ - Полное руководство", и, к сожалению, здесь много кода. Так что я бы использовал буст is_enum
, Соедини это, это могло бы быть похоже на
template <typename T>
typename boost::disable_if< boost::is_enum<T>, bool>::type
ConvertString(const std::string& theString, T& theResult)
{
std::istringstream iss(theString);
return !(iss >> theResult).fail();
}
template <typename T>
typename boost::enable_if< boost::is_enum<T>, bool>::type
ConvertString(const std::string& theString, T& theResult)
{
typedef typename get_etype<sizeof find_etype(theResult)>::type
safe_type;
std::istringstream iss(theString);
safe_type temp;
const bool isValid = !(iss >> temp).fail();
theResult = static_cast<T>(temp);
return isValid;
}
Надеюсь это поможет.
И чтобы просто "завершить" вопрос, в C++0x мы можем просто сделать это:
typedef typename std::underlying_type<T>::type safe_type;
Вместо Йоханнеса get_etype
трюк.