kfm*_*e04 14 c++ templates type-traits c++11
我试图返回int64_tif std::is_integral<>::value是真的.
否则,我想打电话to_int64t()给对象.
我在下面的尝试失败,因为不允许部分特殊化功能模板.
码
#include <type_traits>
#include <cstdint>
template<class T,bool is_integral_type>
int64_t to_int64t( const T& t )
{
return t;
}
template<class T>
int64_t to_int64t<T,std::is_integral<T>::value>( const T& t )
{
return t;
}
template<class T>
int64_t to_int64t<T,!std::is_integral<T>::value>( const T& t )
{
return t.to_int64t();
}
int main()
{
int64_t i = 64;
auto x = to_int64t( i );
}
Run Code Online (Sandbox Code Playgroud)
And*_*owl 29
函数模板不能部分专业化,一般来说,使用函数模板特化不是一个好主意.
实现所需的一种方法是使用一种称为标签调度的技术,它基本上包括提供一个转发器函数,该函数根据额外伪参数的值选择正确的重载:
#include <type_traits>
#include <cstdint>
template<class T>
int64_t to_int64t( const T& t, std::true_type )
{
return t;
}
template<class T>
int64_t to_int64t( const T& t, std::false_type )
{
return t.to_int64t();
}
template<class T>
int64_t to_int64t( const T& t )
{
return to_int64t(t, std::is_integral<T>());
}
int main()
{
int64_t i = 64;
auto x = to_int64t( i );
}
Run Code Online (Sandbox Code Playgroud)
另一种可能性是使用基于经典的SFINAE技术std::enable_if.这就是它的样子(注意,因为C++ 11,允许在函数模板上使用默认模板参数):
#include <type_traits>
#include <cstdint>
template<class T, typename std::enable_if<
std::is_integral<T>::value>::type* = nullptr>
int64_t to_int64t( const T& t )
{
return t;
}
template<class T, typename std::enable_if<
!std::is_integral<T>::value>::type* = nullptr>
int64_t to_int64t( const T& t )
{
return t.to_int64t();
}
int main()
{
int64_t i = 64;
auto x = to_int64t( i );
}
Run Code Online (Sandbox Code Playgroud)
另一种可能性,虽然更详细,但是在detail命名空间中定义辅助类模板(可以部分专用)并提供全局转发器 - 我不会将此技术用于此用例,但我展示它因为它可能会来方便的相关设计情况:
#include <type_traits>
#include <cstdint>
namespace detail
{
template<class T, bool = std::is_integral<T>::value>
struct helper { };
template<class T>
struct helper<T, true>
{
static int64_t to_int64t( const T& t )
{
return t;
}
};
template<class T>
struct helper<T, false>
{
static int64_t to_int64t( const T& t )
{
return t.to_int64t();
}
};
}
template<class T>
int64_t to_int64t( const T& t )
{
return detail::helper<T>::to_int64t(t);
}
int main()
{
int64_t i = 64;
auto x = to_int64t( i );
}
Run Code Online (Sandbox Code Playgroud)
你可以使用std::enable_if:
template<class T, typename std::enable_if<std::is_integral<T>::value, int>::type = 0>
int64_t to_int64t( const T& t )
{
return t;
}
template<class T, typename std::enable_if<!std::is_integral<T>::value, int>::type = 0>
int64_t to_int64t( const T& t )
{
return t.to_int64t();
}
Run Code Online (Sandbox Code Playgroud)