考虑以下示例:
struct Scanner
{
template <typename T>
T get();
};
template <>
string Scanner::get()
{
return string("string");
}
template <>
int Scanner::get()
{
return 10;
}
int main()
{
Scanner scanner;
string s = scanner.get<string>();
int i = scanner.get<int>();
}
Scanner
类用于从某些来源提取令牌。上面的代码运行正常,但是当我尝试get
其他整数类型(例如char
或unsigned int
时失败。读取这些类型的代码与读取int
的代码完全相同。我可以复制我想要阅读的所有其他整数类型的代码,但我宁愿为所有整数类型定义一个函数模板。
我尝试了以下内容:
struct Scanner
{
template <typename T>
typename enable_if<boost::is_integral<T>, T>::type get();
};
这就像魅力一样,但我不确定如何让Scanner::get<string>()
再次发挥作用。那么,我如何编写代码以便我可以scanner.get<string>()
和scanner.get<any integral type>()
并且只有一个定义来读取所有整数类型?
更新:红利问题:如果我想根据某些特征接受多个课程范围怎么办?例如:如果我想要有三个get
函数分别接受(i)整数类型(ii)浮点类型(iii)字符串,我应该如何处理这个问题。
答案 0 :(得分:10)
struct Scanner
{
template <typename T>
typename boost::enable_if<boost::is_integral<T>, T>::type get()
{
return 10;
}
template <typename T>
typename boost::disable_if<boost::is_integral<T>, std::string>::type get()
{
return "string";
}
};
更新“如果我想根据某些特征接受多个类别,该怎么办?”
struct Scanner
{
template <typename T>
typename boost::enable_if<boost::is_integral<T>, T>::type get()
{
return 10;
}
template <typename T>
typename boost::enable_if<boost::is_floating_point<T>, T>::type get()
{
return 11.5;
}
template <typename T>
std::string get(
typename boost::disable_if<boost::is_floating_point<T>, T>::type* = 0,
typename boost::disable_if<boost::is_integral<T>, T>::type* = 0)
{
return std::string("string");
}
};
答案 1 :(得分:3)
按照另一个模板。以下是您想要的一般模式:
template <typename T, bool HasTrait = false>
struct scanner_impl;
template <typename T>
struct scanner_impl
{
// Implement as though the trait is false
};
template <typename T>
struct scanner_impl<true>
{
// Implement as though the trait is true
};
// This is the one the user uses
template <typename T>
struct scanner : scanner_impl<T, typename has_my_trait<T>::value>
{
};