如何识别模板参数是否为std::complex
?我想要一个支持所有数据类型的通用方法,如float,double,int等。
我知道使用std :: is_same我可以专门检查给定的类型,例如std::complex<float>
。
但在这里我需要一个通用的方法。
答案 0 :(得分:10)
这可以使用部分模板专业化来完成。
首先定义一个默认为false的全能模板:
template<typename T>
struct is_complex_t : public std::false_type {};
然后为符合条件的类型提供重载:
template<typename T>
struct is_complex_t<std::complex<T>> : public std::true_type {};
我也想添加一个实用功能:
template<typename T>
constexpr bool is_complex() { return is_complex_t<T>::value; }
编辑:对于c ++ 14及更高版本,此实用程序函数不需要或无用,因为std :: integral_type实现了operator()。
用法:
bool int_is_complex = is_complex<int>(); //false
bool complex_is_complex = is_complex<std::complex<float>>(); //true
答案 1 :(得分:4)
您可以将解决方案基于标签调度技术 它遵循一个最小的工作示例:
#include<complex>
#include<utility>
#include<iostream>
class C {
template<typename T>
void f(int, std::complex<T>) {
std::cout << "complex" << std::endl;
}
template<typename T>
void f(char, T &&t) {
std::cout << "something else" << std::endl;
}
public:
template<typename T>
void f(T &&t) {
f(0, std::forward<T>(t));
}
};
int main() {
C c;
c.f(0);
c.f(std::complex<float>{});
}
这里有一个通用方法f
,几乎可以接受所有内容并在内部调度到正确的函数。
答案 2 :(得分:3)
正如我理解你的问题,你正在寻找一种通用方法的实现,以测试给定类型是否是给定模板模板类型的特化。这可以使用类模板来完成,有点像Frank's answer。我将向您介绍专业化的补充方法 - 带有函数重载的模板类型别名:
#include <type_traits>
#include <complex>
#include <iostream>
template <template <class...> class TT, class... Args>
std::true_type is_tt_impl(TT<Args...>);
template <template <class...> class TT>
std::false_type is_tt_impl(...);
template <template <class...> class TT, class T>
using is_tt = decltype(is_tt_impl<TT>(std::declval<typename std::decay<T>::type>()));
int main() {
static_assert(is_tt<std::complex, std::complex<int>>::value, "!");
static_assert(is_tt<std::complex, std::complex<float>>::value, "!");
static_assert(!is_tt<std::complex, float>::value, "!");
}
您可以按如下方式使用该特征:
#include <type_traits>
#include <complex>
#include <iostream>
//complementary approach to specialization one would be to use function overloading
template <template <class...> class TT, class... Args>
std::true_type is_tt_impl(TT<Args...>);
template <template <class...> class TT>
std::false_type is_tt_impl(...);
template <template <class...> class TT, class T>
using is_tt = decltype(is_tt_impl<TT>(std::declval<typename std::decay<T>::type>()));
template <class T>
typename std::enable_if<is_tt<std::complex, T>::value>::type print(T t) {
std::cout << "(" << t.real() << "," << t.imag() << ")" << std::endl;
}
template <class T>
typename std::enable_if<!is_tt<std::complex, T>::value>::type print(T t) {
std::cout << t << std::endl;
}
int main() {
print(std::complex<int>(1, 2));
print(std::complex<double>(1.5, 2.5));
print(5.5);
}
(1,2)
(1.5,2.5)
5.5
答案 3 :(得分:0)
这也是可行的: Live Demo
#include <boost/type_traits/is_complex.hpp>
#include <iostream>
#include <complex>
int main() {
std::cout << std::boolalpha;
std::cout << boost::is_complex<std::complex<float>>::value << "\n";
std::cout << boost::is_complex<long double>::value << "\n";
}