C++14将具有可根据返回值推断其返回类型的函数。
auto function(){
return "hello world";
}
我可以通过返回类型习惯用法将这种行为应用于使用enable_if作为SFINAE的函数吗?
例如,让我们考虑以下两个功能:
#include <type_traits>
#include <iostream>
//This function is chosen when an integral type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_integral<T>::value>::type {
std::cout << "integral" << std::endl;
return;
}
//This function is chosen when a floating point type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_floating_point<T>::value>::type{
std::cout << "floating" << std::endl;
return;
}
int main(){
function(1); //prints "integral"
function(3.14); //prints "floating"
}
如您所见,使用SFINAE按返回类型惯用法选择正确的函数。
但是,这些都是无效功能。 enable_if
的第二个参数默认设置为void
。这将是相同的:
//This function is chosen when an integral type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_integral<T>::value, void>::type {
std::cout << "integral" << std::endl;
return;
}
//This function is chosen when a floating point type is passed in
template<class T >
auto function(T t) -> typename std::enable_if<std::is_floating_point<T>::value, void>::type{
std::cout << "floating" << std::endl;
return;
}
我可以对这两个函数做些什么,以便它们的返回类型由返回值推断出来吗?
gcc 4.8.2(使用--std=c++1y
)
答案 0 :(得分:11)
std::enable_if
不必在返回类型中,从C ++ 11开始,它可以是模板参数的一部分。
所以你的等效函数可以是(或者,这样的效果):
enum class enabler_t {};
template<typename T>
using EnableIf = typename std::enable_if<T::value, enabler_t>::type;
//This function is chosen when an integral type is passed in
template<class T, EnableIf<std::is_integral<T>>...>
auto function(T t) {
std::cout << "integral" << std::endl;
return;
}
//This function is chosen when a floating point type is passed in
template<class T, EnableIf<std::is_floating_point<T>>...>
auto function(T t) {
std::cout << "floating" << std::endl;
return;
}
它也可以是函数中的参数:
//This function is chosen when an integral type is passed in
template<class T>
auto function(T t, EnableIf<std::is_integral<T>>* = nullptr) {
std::cout << "integral" << std::endl;
return;
}
//This function is chosen when a floating point type is passed in
template<class T>
auto function(T t, EnableIf<std::is_floating_point<T>>* = nullptr) {
std::cout << "floating" << std::endl;
return;
}
这将保留自动类型扣除和SFINAE。
答案 1 :(得分:6)
std::enable_if
可以是返回类型,函数参数或模板参数。如果使用返回类型或模板参数,则会出现函数重定义错误,因此需要使用std::enable_if
作为函数参数:
#include <type_traits>
#include <iostream>
template<class T, typename = typename std::enable_if<std::is_integral<T>::value, void>::type>
auto function(T t, typename std::enable_if<std::is_integral<T>::value, void>::type* dummy = nullptr) {
std::cout << "integral" << std::endl;
return 0;
}
//This function is chosen when a floating point type is passed in
template<class T, typename = typename std::enable_if<std::is_floating_point<T>::value, void>::type>
auto function(T t, typename std::enable_if<std::is_floating_point<T>::value, void>::type* dummy = nullptr) {
std::cout << "floating" << std::endl;
return 0.0f;
}
int main()
{
auto ret = function(0); // integral
auto ret2 = function(0.0f); // floating
std::cout << std::boolalpha << std::is_integral<decltype(ret)>::value << std::endl; // true
std::cout << std::is_floating_point<decltype(ret2)>::value << std::endl; // true
}
答案 2 :(得分:3)
在@ user1508519的答案行中,我们可以从方法中删除enable_if参数,并仅将其保留为模板参数。
我们依赖enable_if<false>
未定义type
这一事实,因此enable_if<false>::type
(不存在)是SFINAE的一个很好的工具 - 在方法签名中,包括模板参数。
无需在方法本身中实际使用此模板参数!
因此:
template<class T,
typename std::enable_if<std::is_integral<T>::value>::type* dummy = nullptr>
auto function(T t) {
std::cout << "integral" << std::endl;
return 0;
}
// This function is chosen when a floating point type is passed in
template<class T,
typename std::enable_if<std::is_floating_point<T>::value>::type* dummy = nullptr>
auto function(T t) {
std::cout << "floating" << std::endl;
return 0.0f;
}
int main() {
auto ret = function(0); // integral
auto ret2 = function(0.0f); // floating
cout << std::boolalpha;
cout << std::is_integral<decltype(ret)>::value << endl; // true
cout << std::is_floating_point<decltype(ret2)>::value << endl; // true
}
integral
floating
true
true
答案 3 :(得分:0)
正如其他地方所提到的,std::enable_if
可以用来形成一个返回类型;一个函数参数;或模板参数。
然而,后两种方法的优点在于它们改变了相关功能或对象的签名。另一方面,在返回类型中使用std::enable_if
会使函数和模板参数计数保持不变。
使用C ++ 11中的lambda返回类型的自动推断(可能扩展到C ++ 14中的普通函数),如果有一种技术可以推断返回类型,那将是理想的。 和在返回类型上使用std::enable_if
。让你的蛋糕吃 - 几乎。唉,目前看来这是不可能的。