Effective C++学习笔记(第七章)

条款41:了解隐式接口和编译器多态
  • 对于class而言,接口是显式的,动态通过virtual函数实现,发生于运行期间。
  • 对于template而言,接口是隐式的,通过模板实例化和函数重载解析实现,发生于编译期。
条款42:了解typename的双重意义
  • 声明template参数时,typename和class可以互换。
  • 需要使用typename关键字表示嵌套从属类型名称,如下示例代码:
#include <iostream>
#include <vector>
template <typename T>
void Print2nd(const T& container)
{
    if (container.size() >= 2) {
        typename T::const_iterator iter(container.begin()); //这里必须加typename表示T::const_iterator是一个名称
        iter++;
        int value = *iter;

        std::cout << value << std::endl;
    }
}

其中T::const_iterator就是嵌套从属类型,必须使用typename关键字。

条款43:学习处理模板化基类内的名称

考察以下示例代码:

#include <iostream>

template <typename T>
class B {
public:
    void Foo() {std::cout << "B::Foo" << std::endl;}
};

template <typename T>
class D : public B<T> {
public:
    void Test() {B<T>::Foo();} // 这里必须在Foo()名称前显式指明使用基类名称B<T>
};
int main()
{
    D<int> obj;

    obj.Test();
    return 0;
}

继承一个模板基类时,如果在子类的成员函数中调用基类中的函数,则必须使用“this->”或者“基类名称::”,显式的指定函数是来自基类的,否则编译器不知道该函数来自哪里,相当于显式的触发基类模板的实例化。

条款44:与参数无关的代码抽离出template

主要作用是防止模板实例化后代码膨胀问题。

  • template代码不应该与某个造成膨胀的template参数产生相依关系。
  • 可以使用函数参数或者类成员变量替换template参数。
条款45:成员函数模板接受所有兼容类型

比如C++11中的智能共享指针,在实现基类和子类对应的智能指针转换时即是这种情况,如下所示:

#include <iostream>
#include <memory>

template <typename T>
class SmartPtr {
public:
    SmartPtr<T>(T* p) : ptr(p) {}

    template <typename C>
    SmartPtr<T>(const SmartPtr<C>& other) {ptr = (T*)(other.get());}

    T* get() const {return ptr;}
private:
    T* ptr;
};
class B {
public:
    virtual ~B() {}
};
class D : public B {

};

int main()
{
    SmartPtr<D> sp(new D());

    SmartPtr<B> sp1 = sp;

    return 0;
}

对于“SmartPtr<B> sp1 = sp;”,它调用的是SmartPtr<B>的拷贝构造函数,参数是SmartPtr<D>类型。

条款46:需要类型转换时请为模板定义非成员函数

参考以下示例代码:

  • 版本1:在模板类内定义operator成员
#include <iostream>

template <typename T>
class R {
public:
    R(T x) : val_(x) {}

    R<T> operator*(const R<T>& rhs) {
        int value = this->val_ * rhs.val_;
        return R(value);
    }

    void Print() {std::cout << val_ << std::endl;}

public:
    T val_;
};

int main()
{
    R<int> r1(10), r2(12);
    R<int> res1 = r1 * r2;
    res1.Print(); // 编译OK

    R<int> res2 = r1 * 2;
    res2.Print(); // 编译OK

    R<int> res3 = 2 * r2;
    res3.Print(); // 编译不通过

    return 0;
}

在这个版本中我们定义了一个模板类内的operator,可以看到当编译"2 * r2"这条语句时,编译不通过。因为编译器遇到2这个常量时,它无法知道将它转换为一个R实例对象。所以,必须定义一个非成员函数。

  • 版本2:在类模板外定义一个非成员operator。
#include <iostream>

template <typename T>
class R {
public:
    R(T x) : val_(x) {}
    void Print() {std::cout << val_ << std::endl;}
public:
    T val_;
};

template <typename T>
R<T> operator*(const R<T>& lhs, const R<T>& rhs)
{
    return R<T>(lhs.val_ * rhs.val_);
}

int main()
{
    R<int> r1(10), r2(12);
    R<int> res1 = r1 * r2;
    res1.Print(); // 编译OK

    R<int> res2 = r1 * 2; 
    res2.Print();  // 编译不通过

    R<int> res3 = 2 * r2;
    res3.Print(); // 编译不通过

    return 0;
}

可以看到这次r1 * 2也编译不通过了,这是因为模板函数不接受任何隐式的转换。

  • 版本3:在模板类内定义一个friend函数
#include <iostream>

template <typename T>
class R {
public:
    R(T x) : val_(x) {}
    void Print() {std::cout << val_ << std::endl;}

    friend R<T> operator*(const R<T>& lhs, const R<T>& rhs)
    {
        return R<T>(lhs.val_ * rhs.val_);
    }
public:
    T val_;
};

int main()
{
    R<int> r1(10), r2(12);
    R<int> res1 = r1 * r2;
    res1.Print(); // 编译OK

    R<int> res2 = r1 * 2;
    res2.Print();  // 编译OK

    R<int> res3 = 2 * r2;
    res3.Print(); // 编译OK

    return 0;
}
条款47:请使用traits class表现类型信息

一句话总结:traits class是“type of type”,我们常用的std::is_pod就是这类class,它是编译器执行的。

#include <iostream>
#include <type_traits>

int main()
{
    int a;
    std::cout << std::is_pod<decltype(a)>::value << std::endl;
    return 0;
}
条款48:认识template元编程
  • 模板元编程(TMP)可以将工作由运行期移至编译期,可以得到更高的执行效率。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容