美文网首页
Effective C++学习笔记(第七章)

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

作者: crazyhank | 来源:发表于2022-05-03 13:03 被阅读0次
条款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)可以将工作由运行期移至编译期,可以得到更高的执行效率。

相关文章

网友评论

      本文标题:Effective C++学习笔记(第七章)

      本文链接:https://www.haomeiwen.com/subject/hvgvyrtx.html