C++:替代"std::is_fundamental"?
c++: alternative to 'std::is_fundamental'?
在模板类中的函数中,我试图区分基本类型和其他类型。
在c++ 11中可以这样做:
if(std::is_fundamental<T>::value)
{
// Treat it as a primitive
}
else
{
//Treat it otherwise
}
如果我错了,请纠正我,这不仅仅是在c++ 11中。
在早期版本的c++中是否有替代方法?
你可以在c++ 03中像这样使用Boost的类型特征:
#include <boost/type_traits/is_fundamental.hpp>
...
if(boost::is_fundamental<T>::value)
{
// Treat it as a primitive
}
else
{
//Treat it otherwise
}
使用这段代码您可能会遇到麻烦。如果您需要区分不同的类型特征,这必须在编译时完成,而不是在运行时。根据您执行的操作,if
的两个分支中的一个可能无法编译。因此,最好转发到专门的函数:
void operation_impl(boost::true_type /*other params*/) {
// Treat it is primitive
}
void operation_impl(boost::false_type /*other params*/) {
// Treat it otherwise
}
template<class T>
void operation(/* params*/) {
operation_impl(boost::is_fundamental<T>::type() /*other params*/);
}
使用这种实现技术,只有被使用的分支需要编译(即正确)。
编辑:
这里有一些附加信息。这个问题的解决方案必须与模板的实例化有关。我从is_fundamental
切换到is_array
,以显示操作可能失败的原因。
template <class T>
void fun(T t) {
if(boost::is_array<T>::value)
{
std::cout << "true" << std::endl;
}
else
{
std::cout << "false" << std::endl;
}
}
void f(int i) {
fun(i);
}
它将编译并运行,并且编译器将看到只使用if语句的一个分支,并将另一个分支作为未使用的代码删除。
在我的第二个例子中,我将在使用数组操作的情况下做一些事情:
template<class T>
void fun(T& t) {
if(boost::is_array<T>::value)
{
std::cout << t[0];
}
else
{
std::cout << t;
}
}
void f(int i) {
fun(i);
}
现在它将无法编译。原因是使用int作为模板参数t[0]
是错误的。你不能在编译时使用这个运行时语句来区分代码中需要的类型属性(在这个例子中是数组属性和t[0]
的使用)。
在第三个例子中,我们将通过函数重载在编译时进行区分:
template<class T>
void fun_impl(boost::true_type, T& t) {
std::cout << t[0];
}
template<class T>
void fun_impl(boost::false_type, T& t) {
std::cout << t;
}
template<class T>
void fun(T& t) {
fun_impl(typename boost::is_array<T>::type(),t);
}
void f(int i) {
fun(i);
}
这里is_array<T>::type
是true_type
或false_type
。该结果用作选择器,用于在编译时选择正确的fun_impl
重载,并且只有被选择的重载才会被实例化和编译。
通常,这些技术用于在编译时选择一个最佳实现,该实现可能仅在类型具有某些属性时才可编译。
编辑2:
如果static if
是语言的一部分,这当然会改变。
相关文章:
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 如何导出包含具有"std::unique_ptr"值的"std::map"属性的
- 从持续时间构造std::chrono::system_clock::time_point
- std::具有相同基类的类的变体
- std::向量与传递值的动态数组
- 使用std::vector的OpenCL矩阵乘法
- std::map<struct,struct>::find 找不到匹配项,但是如果我循环通过 begin() 到 end(),我在那里看到匹配项
- std::condition_variable::wait()如何评估给定的谓词
- 如何获取std::result_of函数的返回类型
- std::原子加载和存储都需要吗
- 将对象移动到std::shared_ptr
- POCO::PostgreSQL:如何将std::vector支持添加到`Binder::bind`
- 使用一个考虑到std::map中键值的滚动或换行的键
- 如何从 std::atomic 中提取指针 T<T>?
- 为什么 std::unique 不调用 std::sort?
- 使用std::函数映射对象方法
- 可组合的lambda/std::函数与std::可选