C++中的Template常见用法

59 篇文章 0 订阅
52 篇文章 2 订阅

目录

摘要

函数模板

基本用法

重载与特化

类模板

基本用法

模板特化(Specialization)与偏特化(Partial Specialization)

别名模板(Alias Templates)

基本用法

变量模板(Variable Templates)

基本用法

模板模板参数(Template Template Parameters)

基本用法

总结


摘要

C++中的模板(template)是一种非常强大的工具,用于泛型编程和代码复用。模板可以分为函数模板、类模板、别名模板、变量模板、模板模板参数等。这些模板能够处理任意类型的数据,提高代码的灵活性和可重用性。

函数模板

函数模板允许函数对任意类型的数据进行操作。

基本用法

#include <iostream>

template <typename T>
T max(T a, T b) {
    return (a > b) ? a : b;
}

int main() {
    std::cout << "Max of 3 and 7: " << max(3, 7) << std::endl;
    std::cout << "Max of 3.5 and 2.1: " << max(3.5, 2.1) << std::endl;
    std::cout << "Max of 'a' and 'z': " << max('a', 'z') << std::endl;
    return 0;
}

重载与特化

函数模板可以与普通函数重载,并且可以进行模板特化。

#include <iostream>

template <typename T>
T max(T a, T b) {
    return (a > b) ? a : b;
}

// 特化版本
template <>
const char* max(const char* a, const char* b) {
    return (std::strcmp(a, b) > 0) ? a : b;
}

// 重载版本
int max(int a, int b) {
    return (a > b) ? a : b;
}

int main() {
    std::cout << "Max of 3 and 7: " << max(3, 7) << std::endl;
    std::cout << "Max of \"hello\" and \"world\": " << max("hello", "world") << std::endl;
    return 0;
}


// Out
Max of 3 and 7: 7
Max of "hello" and "world": world

类模板

类模板允许类处理任意类型的数据。

基本用法

#include <iostream>

template <typename T>
class Pair {
private:
    T first, second;
public:
    Pair(T a, T b) : first(a), second(b) {}
    T getFirst() const { return first; }
    T getSecond() const { return second; }
};

int main() {
    Pair<int> intPair(1, 2);
    Pair<double> doublePair(3.4, 7.8);

    std::cout << "First of intPair: " << intPair.getFirst() << std::endl;
    std::cout << "Second of doublePair: " << doublePair.getSecond() << std::endl;

    return 0;
}

模板特化(Specialization)与偏特化(Partial Specialization)

类模板可以进行全特化(Full Specialization)和偏特化(Partial Specialization -- 偏特化仅在类模板中支持)。

#include <iostream>

// 主模板
template <typename T>
class Storage {
private:
    T value;
public:
    Storage(T val) : value(val) {}
    void print() const { std::cout << "Value: " << value << std::endl; }
};

// 全特化版本
// 全特化是为模板的某个具体类型提供完全不同的实现。
template <>
class Storage<const char*> {
private:
    const char* value;
public:
    Storage(const char* val) : value(val) {}
    void print() const { std::cout << "Value: " << value << std::endl; }
};

// 偏特化版本
// 偏特化是指为模板的一部分参数提供特定的实现,而不是所有参数。偏特化仅适用于类模板,而不适用于函数模板。
template <typename T>
class Storage<T*> {
private:
    T* value;
public:
    Storage(T* val) : value(val) {}
    void print() const { std::cout << "Pointer Value: " << *value << std::endl; }
};

int main() {
    Storage<int> intStorage(5);
    Storage<const char*> stringStorage("Hello");
    int value = 10;
    Storage<int*> intPointerStorage(&value);

    intStorage.print();
    stringStorage.print();
    intPointerStorage.print();

    return 0;
}


// Out
Value: 5
Value: Hello
Pointer Value: 10

别名模板(Alias Templates)

别名模板用于创建复杂类型的简化别名。

基本用法

#include <iostream>
#include <vector>

// 定义一个别名模板
template <typename T>
using Vec = std::vector<T>;

int main() {
    Vec<int> intVec = {1, 2, 3, 4, 5};
    
    for (int i : intVec) {
        std::cout << i << " ";
    }
    std::cout << std::endl;

    return 0;
}

变量模板(Variable Templates)

变量模板允许定义模板变量。

基本用法

#include <iostream>

// 定义一个变量模板
template <typename T>
constexpr T pi = T(3.1415926535897932385);

int main() {
    std::cout << "pi<int>: " << pi<int> << std::endl;
    std::cout << "pi<float>: " << pi<float> << std::endl;
    std::cout << "pi<double>: " << pi<double> << std::endl;

    return 0;
}


// Out
pi<int>: 3
pi<float>: 3.14159
pi<double>: 3.14159

模板模板参数(Template Template Parameters)

模板模板参数允许模板参数本身是模板。

基本用法

#include <iostream>
#include <vector>

// 定义一个模板类
template <typename T, template <typename, typename> class Container>
class Stack {
private:
    Container<T, std::allocator<T>> elements;
public:
    void push(const T& elem) {
        elements.push_back(elem);
    }

    void pop() {
        if (elements.empty()) {
            throw std::out_of_range("Stack<>::pop(): empty stack");
        }
        elements.pop_back();
    }

    T top() const {
        if (elements.empty()) {
            throw std::out_of_range("Stack<>::top(): empty stack");
        }
        return elements.back();
    }
};

int main() {
    Stack<int, std::vector> intStack;
    intStack.push(7);
    intStack.push(42);
    
    std::cout << "Top of intStack: " << intStack.top() << std::endl;
    intStack.pop();
    std::cout << "Top of intStack after pop: " << intStack.top() << std::endl;

    return 0;
}


// Out
Top of intStack: 42
Top of intStack after pop: 7

总结

1. 模板实例化
模板实例化发生在使用模板时,编译器会根据模板参数生成具体类型的代码。模板实例化可能会导致代码膨胀(code bloat),因为每个不同的模板参数都会生成一份独立的代码。

2. 模板元编程
模板元编程是利用模板在编译时进行计算和逻辑判断的技术,常用于实现复杂的数据结构和算法。

3. 编译错误
模板代码的编译错误信息通常较难理解,因为错误通常发生在模板实例化的过程中。

4. 类型推导
模板参数的类型推导有时会遇到困难,特别是在模板函数中。

5. 递归模板实例化
在某些高级用法中,模板可能会涉及递归实例化(如元编程)。这种情况下,需要确保递归在某个点终止,以避免编译时无限递归。

我们可以通过合理使用函数模板、类模板、别名模板、变量模板和模板模板参数,能在一定程度上提高代码的灵活性和可维护性。然而,模板的使用也带来了一些复杂情况和注意事项,如编译时间、代码膨胀、调试困难等。所以在实际项目中,根据项目需求,我们需要合理设计和选择模板,这样能够有效地避免问题的发生。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

沉夢志昂丶

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值