一个专注音视频领域问答的小圈子

现如今,掌握 C++ 模板技巧并且熟练使用可以说是能力进阶的必备内容了。

在一些优秀的开源项目中经常能看到模板的使用,要是不了解其使用方法,对分析源码都会有些阻碍。

推荐阅读《C++ Templates 中文版》一书,或许可以让你对 C++ 模板有个更加系统的概念,同时辅助阅读网上相关的博客文章加深理解,在代码实践中去掌握提高。

C++ 模板主要可以分为函数模板和类模板,这次就是介绍它们两个。

函数模板

函数模板主要就是将它的参数用类型带代替,对每个不同的类型都会生成对应的函数,所以函数模板可以用多种不同的类型进行调用,它代表的其实是一个函数家族而不是一个函数。

以下是个简单的函数模板示例:

#include <iostream>

template <typename T>
T plus(T a,T b){
    return a+b;
}

// 使用引用类型,减少拷贝操作
template<typename T>
inline T const &max(T const &a, T const &b) {
    return a < b ? b : a;
}

int main() {
    plus(1.0, 1.0);
    plus(static_cast<int>(1.0), 2);
    plus<int>(2.0, 3.0);

    max(2,1);
    max(static_cast<int>(2.0),1);
    max<double>(2.0,3.0);
    return 0;
}

通常而言,并不是把模板编译成一个可以处理任何单一类型的单一实体,而是针对实例模板参数的每种类型,都从模板产生一个不同的实体。

针对上面的使用,整型和浮点型都生成了一个对应的函数。

对于函数模板的参数,是可以在调用时显示指定的,另外可以在调用时对类型做强制类型转换,这是因为模板不允许自动类型转换,所以需要我们手动来完成。

函数模板的重载

和普通函数一样,函数模板也可以重载,也就是相同函数名称可以具有不同的函数定义。

inline int const &max(int const &a, int const &b) {
    return a < b ? b : a;
}

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

template<typename T>
inline T const &max(T const &a, T const &b, T const &c) {
    return ::max(::max(a,b),c);
}

如上所示,一个非函数模板可以合一个同名的函数模板同时存在,而且该函数模板还可以被实例化为这个非函数模板。

对于非函数模板和同名的函数模板,如果其他条件都相同的话,那么在调用的时候,重载解析过程通常会调用非函数模板,而不会从该模板产生一个实例。

对于函数的重载,一般来说改变的内容最好是下面两种情况:

  • 改变参数的数目
  • 显示地指定模板参数

遵循以上两点,避免在重载函数调用过于复杂,从而导致匹配出差。

类模板

与函数模板相似,类可以被一种或者多种类型参数化。

以下代码是一个简单的类模板示例:

template<typename T>
class Stack {
private:
    std::vector<T> elems;
public:
        void push(T const&);
        void pop();
        T top() const;
        bool empty() const{
            return elems.empty();
        }
};

template<typename T>
void Stack<T>::push(const T & elem) {
    elems.push_back(elem);
}

template<typename T>
void Stack<T>::pop() {
    if (!elems.empty()){
        elems.pop_back();
    }
}

template<typename T>
T Stack<T>::top() const {
    if (!elems.empty()){
        return elems.back();
    }
    throw std::out_of_range("Stack::top(); empty Stack");
}

int main() {
    Stack<int> intStack;
    intStack.push(1);
    intStack.top();
    intStack.pop();
    return 0;
}

类模板的声明和函数模板声明很类似,在声明之前都用 typename 声明作为类型参数的标识符。

在定义类模板的成员函数时,关于成员函数的实现,可以在类声明里面去实现,也可以放到外面去实现。

如果是在类外面声明的,那么必现要指定该成员函数是一个函数模板,而且还需要使用这个类模板的完整类型限定符。

另外,只有那些被调用的成员函数,才会产生这些函数的实例化代码。

类模板的特化

可以使用模板实参来特化类模板。

和函数模板的重载类似,通过特化类模板,可以优化基于某种特定类型的实现,或者克服某种特定类型在实例化类模板时所出现的不足。

另外,如果要特化一个类模板,还要特化该类模板的所有成员函数。虽然也可以只特化某个成员函数,但这个做法并没有特化整个类,也就没有特化整个类模板。

为了特化一个类模板,必须在起始处声明一个 template<> ,接下来声明用特化类模板的类型。

如下代码所示:

template<>
class Stack<std::string> {
private:
        std::deque<std::string> elems;
public:
        void push(std::string const&);
        void pop();
        std::string top() const;
        bool empty() const{
            return elems.empty();
        }
};

void Stack<std::string>::push(const std::string &elem) {
    elems.push_back(elem);
}

void Stack<std::string>::pop() {
    if (!elems.empty()){
        elems.pop_back();
    }
}

std::string Stack<std::string>::top() const {
    if (!elems.empty()){
        return elems.back();
    }
    throw std::out_of_range("Stack<std::string>::top(); empty Stack");
}

int main() {
    Stack<std::string> stringStack;
    intStack.push("string");
    intStack.top();
    intStack.pop();
    return 0;
}

进行类模板的特化时,每个成员函数都必须重新定义为普通函数,原来函数模板中的每个 T 也相应地被进行特化的类型取代。

局部特化

上面的特化可以称之为全特化,另外类模板还可以局部特化,在特定的环境下制定类模板的实现,并且要求某些模板参数仍然必须由用户来定义。

如下代码所示:

template<typename T1,typename T2>
class MyClass{

};

// 局部特化,两个模板参数具有相同的类型
template<typename T>
class MyClass<T,T>{

};

// 局部特化, 第二个模板参数的类型是 int
template<typename T>
class MyClass<T,int>{

};

// 局部特化,两个模板参数都是指针类型
template<typename T1,typename T2>
class MyClass<T1*,T2*>{

};

对于局部特化,其成员函数同样要指定是模板函数,并且要使用特化后的类模板完整类型限定符。

缺省模板实参

对于类模板,可以为类模板参数定义默认值。这些值就被称为缺省模板实参,而且它们还可以引用之前的目标参数。

如下代码所示:

template <typename T,typename CONT = std::vector<T> >
class MyStack{
private:
    CONT elems;
public:
    void push(T const&);
    void pop();
    T top() const;
    bool empty() const{
        return elems.empty();
    }
};

template<typename T, typename CONT>
void MyStack<T, CONT>::push(const T &elem) {
    elems.push_back(elem);
}

template<typename T, typename CONT>
void MyStack<T, CONT>::pop() {
    if (!elems.empty()){
        elems.pop_back();
    }
}

template<typename T, typename CONT>
T MyStack<T, CONT>::top() const {
    if (!elems.empty()){
        return elems.back();
    }
    throw std::out_of_range("MyStack<>::top(); empty Stack");
}

int main() {
    MyStack<int> intMyStack;
    intMyStack.push(1);
    intMyStack.top();
    intMyStack.pop();

    MyStack<double, std::deque<double>> doubleMyStack;
    doubleMyStack.push(0.1);
    doubleMyStack.top();
    doubleMyStack.pop();
    return 0;
}

以上代码中,如果不传递第二个模板参数,那么就使用默认的 vector 来管理元素。如果传了,就使用指定的类型来管理元素。

小结

以上就是关于 C++ 模板中函数模板和类模板的小总结,基本上还是很容易理解的,就是用模板参数 T 代替具体的类型,并且还可以对函数模板进行重载,对类模板进行特化,类模板还可以指定默认类型参数。

原创文章,转载请注明来源:    C++ 模板系列小结01-函数模板和类模板