c++模板--基础知识

在经历前两讲之后,我想更为全面的介绍一下c++模板的基础知识。本讲将假设你拥有上两讲的基础知识了。

1.关键子typename

        在c++标准化过程中,引入关键子typename是为了说明:模板内部的标识符可以是一个类型

 比如:

template<typename T>
class MyClass {
    typename T::SubType* ptr;
    ......
};

 第二个typename表示:SubType是定义于类T内部的一种类型.

如果没有第二个typename的声明, 直接写成下面这样的形式

T::SubType* ptr;

编译器会认为:SubType 是类T里的静态成员变量, 所以上面这段代码会被解释为:

T::SubType与ptr的乘积(T::SubType  *  ptr)

2.  .template构造(注意template前的小点)

先看例子

template<int N>
void printBitset(std::bitset<N> const& bs){
    std::cout<<bs.template to_string<char,char_traits<char>,allocator<char> >();
}

你会发现一个奇怪的东西bs.template.实际上我们是想像下面这么调用

bs.to_string<char,char_traits<char>,allocator<char> >();

bs是一个std::bitset<N> const&类型的对象, 我们想调用这个对象下的to.string...

但是有一个问题, 编译器如何的区分后面的<, >到底是模板的<>,还是比较大小的<, >.

于是用.template构造来告诉编译器, 后面看到的<>,都不是比较大小的<>

3.使用->this

例子

template <typename T>
class Base {
public:
    void exit();
};


template <typename T>
class Derived : Base<T> {
public:
    void foo() {
        exit(); //调用外部的exit()或者直接报错
    }
};

直接调用的话,编译器要么报错,要么调用其他同名的exit()函数,反正不会调用从父类的继承的exit()函数.

如果想要调用从父类继承的函数,应该像下面这么写

this->exit();
//或者
Base<T>::exit();

4.成员模板

模板类里的成员也可以是一个模板

例子

template <typename T>
class Stack {
private:
    std::deque<T> elems;//存储元素的容器

public:
    void push(T const&);//入栈
    void pop();//出栈
    T top() const;//返回栈顶元素
    inline bool empty() const { return elems.empty(); }

    //使用元素类型为T2的栈进行赋值
    template <typename T2>
    Stack<T>& operator= (Stack<T2> const&);
    
}

template<typename T>
inline
template<typename T2>
Stack<T>& Stack<T>::operator= (Stack<T2> const& op2) {
    if((void*)this == (void*)&op2) {//我们不希望自己赋值给自己
        return *this;
    }

    Stack<T2> tmp(op2); //拷贝一个副本

    elems.clear(); //清空已有的元素
    while(!tmp.empty()){
        elems.push_back(tmp.back());
        tmp.pop_back();
    }

    return *this;
}

5.模板的模板参数

比较长,单独写了一篇

6.零初始化

我们希望当我们使用一个模板的时候,即使我们什么都不做,其内部就已经做好了初始化,以免出现一些预料之外的事情。

//对于函数模板
template<typename T>
void f() {
    T x = T();
    //加入T是int,就是int x = int();,这样x会被初始化为0
}


//对于类模板
template<typename T>
class MyClass {
private:
    T x;
public:
    MyClass() : x(...) {//确认x被初始化了,其他内建对象类型也一样
        .....
    }
}

7.使用字符串作为函数模板的实参

有时把字符串传递给函数模板的引用会出现意想不到的事情,这与模板的类型推导和数组的退化有关,如果你还不知道我在说什么的话,可以去看我相关的文章

模板的类型推导

auto的类型推导

举例

#include <string>

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

int main() {
    std::string s();

    ::max("apple", "peach"); //正确,相同的类型参数
    ::max("apple", "tomato");//错误,不同的类型参数
    ::max("apple", s);//错误,不同类型的实参
}

由于是T const&,所以数组不会退化。
那么apple和peach是const char[6]类型
tomato是const char[7]类型
s是一个string类型

比较好的解决方法是为字符串重载max()。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值