泛型编程技巧——使用std::enable_if实现按类型进行条件编译​

std::enable_if可以以多种形式使用,包括:

  • 作为类模板或函数模板参数
  • 作为返回类型(不适用于构造函数和析构函数)
  • 作为附加函数参数(不适用于运算符重载)

std::enable_if can be used in many forms, including:

  • as an additional function argument (not applicable to most operator overloads),
  • as a return type (not applicable to constructors and destructors),
  • as a class template or function template parameter.

std::enable_if 核心是结合 SFINAE(Substitution Failure Is Not An Error)机制,实现编译期的条件分支逻辑。

c++14中,定义了using enable_if_t = typename enable_if<B,T>::type;

1. 作为类模板或函数模板参数

1.1 控制类模板的参数类型

例子,

// the partial specialization of A is enabled via a template parameter
template<class T, class Enable = void>
class A {}; // primary template
 
template<class T>
class A<T, typename std::enable_if<std::is_floating_point<T>::value>::type>
{}; // specialization for floating point types

int main()
{
    A<int>{}; // OK: matches the primary template
    A<double>{}; // OK: matches the partial specialization
}

另一个例子,

// 基础模板
template <typename Mat, typename Vec, typename Enable = void>
class Derived : public Base<Mat, Vec> {};  


// 模版特化,针对double类型
template <typename Mat, typename Vec>
class Derived<Mat, Vec, std::enable_if_t<std::is_same_v<typename Mat::ElementType, std::complex<double>>>> : public Base<Mat, Vec> {};


// 模版特化,针对std::complex<double>类型
template <typename Mat, typename Vec>
class Derived<Mat, Vec, std::enable_if_t<std::is_same_v<typename Mat::ElementType, std::complex<double>>>> : public Base<Mat, Vec> {};

在这个例子中,Derived继承自Base基类,但是能够针对Mat元素类型来进行特化。

这个例子使用了c++14新特性enable_if_t 。

1.2 控制函数模板的参数类型

例如,根据类型是否为整数类型来选择不同的函数实现:

#include <type_traits>
#include <iostream>

template<typename T, typename std::enable_if<std::is_integral<T>::value, T>::type* = nullptr>
void print(T value) {
    std::cout << "Integral value: " << value << std::endl;
}

template<typename T, typename std::enable_if<!std::is_integral<T>::value, T>::type* = nullptr>
void print(T value) {
    std::cout << "Non-integral value: " << value << std::endl;
}

int main() {
    print(5);       // Integral value: 5
    print(3.14);    // Non-integral value: 3.14
    return 0;
}

在这个例子中:

  • 第一个 print 函数模板仅对整数类型有效(std::is_integral<T>::valuetrue)。

  • 第二个 print 函数模板对非整数类型有效(std::is_integral<T>::valuefalse)。

2. 作为返回类型

例子,

#include <type_traits>
#include <iostream>

template <typename T>
typename std::enable_if<std::is_integral<T>::value, T>::type
foo(T x)
{
    std::cout << "foo int";
    return x * 2;
}

template <typename T>
typename std::enable_if<std::is_floating_point<T>::value, T>::type
foo(T x)
{
    std::cout << "foo float";
    return x / 2.0;
}

在这个示例中,我们定义了两个函数模板foo,它们根据不同的类型来返回不同的结果。第一个foo函数只能接受整型参数,并返回这个整数的两倍。第二个foo函数只能接受浮点型参数,并返回这个浮点数的一半。

我们运行

foo(2);
foo(2.1);

将得到打印结果

foo int
foo float

3. 作为附加函数参数

以下是一个使用 std::enable_if 作为附加函数参数的示例,假设我们希望根据传入的类型是否为指针类型来选择不同的函数实现:

#include <iostream>
#include <type_traits>

// 重载函数1:仅对指针类型有效
template<typename T>
void print(T* value, typename std::enable_if<std::is_pointer<T*>::value>::type* = nullptr) {
    std::cout << "Pointer value: " << *value << std::endl;
}

// 重载函数2:对非指针类型有效
template<typename T>
void print(T value, typename std::enable_if<!std::is_pointer<T>::value>::type* = nullptr) {
    std::cout << "Non-pointer value: " << value << std::endl;
}

int main() {
    int a = 10;
    int* ptr = &a;

    print(ptr);  // 调用指针版本的 print
    print(a);    // 调用非指针版本的 print

    return 0;
}

运行程序后,输出结果如下:

Pointer value: 10
Non-pointer value: 10

参考:

https://en.cppreference.com/w/cpp/types/enable_if.html

C++ std::enable_if的简明指南_c enable if-优快云博客

<think>好的,我现在需要回答用户关于`std::is_class`的问题。首先,用户之前问过关于Python中int析构函数的问题,现在转向C++的`std::is_class`,可能是在学习类 traits。 用户的问题是“介绍一下std::is_class”,我需要解释这个模板的用途、工作原理以及使用场景。考虑到用户可能对C++模板和类 traits 不太熟悉,应该用简单明了的语言,结合例子说明。 首先,确定`std::is_class`属于C++标准库中的类特性库,位于<type_traits>头文件。它的作用是检查一个类是否为类类(包括结构体)。但要注意,联合体(union)不被认为是类类在这里,所以`std::is_class`对union会返回false。 接下来需要说明`std::is_class`的用法,比如通过成员value或operator bool来获取结果。可能还需要提到它继承自std::true_type或std::false_type,这样用户知道可以用于模板元编程中的分支。 然后,解释实现原理。这部分可能需要涉及编译器内部机制,因为类 traits 的实现通常依赖于编译器的支持。比如,通过模板特化或编译器内置功能来判断类是否为类。同时,要指出用户通常不需要自己实现,标准库已经提供。 应用场景部分,可以举几个例子,比如在模板中根据类是否为类类来选择不同的实现,或者在序列化、反射库中需要判断类特性时使用。这样用户能理解其实际用途。 需要注意的细节包括:基本类(如int)、枚举、联合体、指针等不被视为类类,而结构体和类则是。此外,可能存在的陷阱,比如cv限定符(const/volatile)是否会影响判断,应该指出`std::is_class`会忽略顶层的cv限定符,但可以配合std::remove_cv使用。 最后,通过代码示例展示如何使用`std::is_class`,包括静态断言和运行时输出,这样用户能直观看到结果。确保例子覆盖结构体、联合体、基本类等不同情况,帮助用户理解不同情况下的输出结果。 总结时要强调`std::is_class`在模板元编程中的作用,帮助开发者根据类特性编写更灵活的代码。同时,提醒用户注意其局限性,比如无法区分类和其他复合类,可能需要结合其他traits使用。 检查是否有遗漏点,比如C++11引入,需要包含哪个头文件,以及与其他相关traits(如std::is_union)的区别。确保语言通俗,避免过于技术化的术语,但保持准确性。</think>`std::is_class` 是 C++ 标准库中的一个类特性(Type Trait),用于在编译时判断一个类是否为**类类**(包括 `class` 或 `struct` 定义的类)。它属于 `<type_traits>` 头文件,常用于模板元编程中根据类特性进行条件分支或静态断言。 --- ### **核心特性** - **作用**:检查类 `T` 是否为类或结构体。 - **返回值**: - 若 `T` 是类或结构体,则 `std::is_class<T>::value` 为 `true`。 - 否则(如基本类、枚举、联合体等),结果为 `false`。 - **继承关系**:派生自 `std::true_type` 或 `std::false_type`,可直接用于编译时逻辑。 --- ### **使用方式** ```cpp #include <type_traits> // 检查类是否为类 static_assert(std::is_class<MyClass>::value, "MyClass 是类类"); static_assert(!std::is_class<int>::value, "int 不是类类"); // 结合模板的编译时分支 template<typename T> void process(T value) { if constexpr (std::is_class_v<T>) { // T 是类类的处理逻辑 } else { // T 是非类类的处理逻辑 } } ``` --- ### **实现原理** `std::is_class` 的实现依赖于编译器的内部机制。其底层逻辑大致如下: 1. **排除非类类**:基本类(如 `int`)、数组、函数、指针、引用等直接被判定为非类类。 2. **识别类类**:编译器通过类系统直接判断 `class` 或 `struct` 定义的类。 3. **特化处理**:标准库通过模板特化实现,例如: ```cpp template<typename T> struct is_class : std::integral_constant<bool, __is_class(T)> {}; ``` 其中 `__is_class` 是编译器内置的类判断函数(如 GCC/Clang 中的内置特性)。 --- ### **应用场景** 1. **模板约束**:在编程中,限制模板参数必须为类类。 ```cpp template<typename T, typename = std::enable_if_t<std::is_class_v<T>>> void serialize(const T& obj) { /* 序列化逻辑 */ } ``` 2. **类分发**:根据类是否为类选择不同实现。 3. **静态断言**:在编译时验证类是否符合预期。 --- ### **注意事项** 1. **联合体(Union)**:`union` 不被视为类类,需使用 `std::is_union` 单独判断。 2. **cv限定符**:`std::is_class` 忽略顶层 `const`/`volatile` 限定符。若需保留限定符,需配合 `std::remove_cv` 使用。 3. **继承类**:基类和派生类均被视为类类。 4. **空类**:空类(无成员变量或函数)也会返回 `true`。 --- ### **示例代码** ```cpp #include <iostream> #include <type_traits> struct MyStruct {}; class MyClass {}; union MyUnion {}; int main() { std::cout << std::boolalpha; std::cout << "int 是类类? " << std::is_class_v<int> << '\n'; // false std::cout << "MyStruct 是类类? "<< std::is_class_v<MyStruct> << '\n'; // true std::cout << "MyClass 是类类? " << std::is_class_v<MyClass> << '\n'; // true std::cout << "MyUnion 是类类? " << std::is_class_v<MyUnion> << '\n'; // false } ``` 输出结果: ``` int 是类类? false MyStruct 是类类? true MyClass 是类类? true MyUnion 是类类? false ``` --- ### **总结** `std::is_class` 是 C++特性工具中的基础组件,通过编译时类检查帮助开发者编写更安全、灵活的模板代码。其核心优势在于**零运行时开销**,结果完全在编译期确定,适用于高性能场景。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值