`

Efficient C++ 第三章

 
阅读更多
转自
http://blog.chinaunix.net/uid-25872711-id-3015683.html

Efficient C++ 第三章


虚函数
  看如下例子:
class ZooAnimal {
public:
    ...
    virtual void draw();
    int resolveType() {return myType;}
private:
    int myType;
    ...
}

    ZooAnimal为动物园所有动物的基类,myType标明类型,如果要画出所有动物,可用如下方式实现:
void drawAllAnimals (ZooAnimal *pz) // pointer to first animal in the
                                    // list
{
    for (ZooAnimal *p=pz; p ;p = p->next) {
         switch (p->resolveType()) {
         case BEAR:
              ( (Bear *) p)->draw();
              break;
         case MONKEY:
              ((Monkey *) p)->draw();
              break;
         ... // Handle all other animals currently in the zoo.
         }
         }
}


    但是这样的代码维护比较烦琐,当新增动物,或者去除动物的时候你需要不断的修改switch语句。这里可以使用虚函数处理,利用其动态绑定。
void drawAllAnimals (ZooAnimal *pz) // pointer to first animal in the
                                    // list
{
    for (ZooAnimal *p=pz; p ;p = p->next) {
         p->draw();
         }
}


    当类X中存在虚函数,那么编译器会为类X产生一个虚函数表,虚函数表拥有该类的所有虚函数的指针。每一个类有一个虚函数表,类的每个对象都有一个隐藏的指向该表的指针。隐藏是因为,只有编译器知道vptr在对象内部的偏移量。编译器在对象的构造函数中插入代码,以正确初始化vptr。
  虚函数似乎有以下的几方面开销:
    1) 必须在构造函数内初始化vptr
    2) 虚函数是通过指针间接调用,必须先得到指向虚函数表的指针,然后访问正确的函数偏移量
    3) 内联是编译时的选择,由于虚函数的类型判断发生在运行时,所以编译器不能内联虚函数。

  公平的来讲,前两项并不算做性能损失。因为即使不使用虚函数,也需要付出类别的开销。初始vptr的开销等价于Bear中初始化类型的开销,如:
class Bear : public ZooAnimal {
    ...
    Bear (const char *name) : myName(name), myType(BEAR) {}
    ...
};

    第二条函数调用的开销,等价于switch逻辑和区分不同类型Bear::Draw()的开销:
switch (p->resolveType()) {
 case BEAR:
   ( (Bear *) p)->draw();
   break;
 case MONKEY:
   ((Monkey *) p)->draw();
   break;
 ... // Handle all other animals currently in the zoo.
 }

    其实,虚函数真正的损失只有第三条,无法内联函数是虚函数最大的性能损失。
  由于函数内联造成的性能损失没有固定的代价,如果是简短、调用频繁的函数那么性能损失就比较明显。可以用模板来解决这一问题。

模板和继承
  看如下线程锁的示例代码,要实现不同线程锁类型CriticalSection和Mutex的线程安全string类:
class Locker {
public:
    Locker() {}
    virtual ~Locker() {}
    virtual void lock() = 0;
    virtual void unlock() = 0;
};

    按以下三种方法设计:
    1)硬编码:从string类派生CirticalSectionString和MutexString,每个类包含特定的锁
    2)虚函数:从string派生一个ThreadSafeString,包含Locker锁基类指针,通过虚函数加锁,开锁。
    3)模板:创建一个Locker参数的模板字符串类

硬编码方式:
class CriticalSectionString : public string {
public:
    ...
    int length();
private:
    CriticalSectionLock cs;
};

int CriticalSectionString::length()
{
    cs.lock()
    int len = string::length();
    cs.unlock();

    return len;
}


    显然这种方式没有额外开销,不同锁机制实现不同的string类型,lock和unlock可以在编译时刻确定,进行内联。

虚函数:
class ThreadSafeString : public string {
public:
    ThreadSafeString (const char *s, Locker *lockPtr)
    : string(s), pLock(lockPtr) {}
    ...
    int length();
private:
    Locker *pLock;
};

int ThreadSafeString::length() 
{
 pLock->lock();
 int len = string::length();
 pLock->unlock();
 return len; 
}

//使用时,如下:
{
    CriticalSectionLock cs;
    ThreadSafeString csString("Hello", &cs);
    ...
}

{
    MutexLock mtx;
    ThreadSafeString csString("Hello", &mtx);
    ...
}

    通过虚函数动态绑定正确类型lock和unlock,代码较简单,但是虚函数不能内联,有性能损失。

模板:
   
template <class LOCKER>
class ThreadSafeString : public string {
public:
    ThreadSafeString(const char *s) : string(s) {}
    ...
    int length();

private:
    LOCKER lock;
};

template <class LOCKER>
inline
int ThreadSafeString<LOCKER>::length()
{
    lock.lock();
    int len = string::length();
    lock.unlock();

    return len;
}

{
    ThreadSafeString <CriticalSectionLock> csString = "hello";
    ...
}

{
    ThreadSafeString <MutexLock> mtxString = "hello";
    ...
}


    模板可以很好的实现多态,也可以在编译时刻确定类型,因此lock和unlock可以内联,这样就没有不必要开销。

要点:
  1)因为虚函数不能内联,这对调用频繁、简单的函数的性能开销影响较大。
  2)如果设计可行,可以用模板,取代虚函数实现多态,减小性能损失。
分享到:
评论

相关推荐

    Effective C++ 3个版本清晰pdf

    - **第二版到第三版**:针对C++11及后续标准进行了更新,涵盖了新的语言特性,如lambda表达式、右值引用和类型推断等。 通过阅读这三个版本的《Effective C++》,程序员可以全面了解C++语言的发展历程,掌握从早期...

    Effective C++ 中文版第三版 高清PDF.pdf

    《Effective C++ 中文版第三版》是一本深入探讨C++编程实践的书籍,由Scott Meyers撰写,旨在帮助读者提升C++编程的效率和专业性。这本书的高清PDF版本提供了一个清晰易读的阅读体验,是C++程序员进阶学习的理想资料...

    Modern and efficient C++ Thread Pool Library.zip

    "Modern and efficient C++ Thread Pool Library"很可能是一个第三方库,旨在提供比标准库更高级、性能更好的线程池实现。这个库可能包含以下关键知识点: 1. **线程池概念**:线程池是一组预先创建的线程,等待...

    C++数值算法第三版源码

    In particular, it is difficult to treat vectors and matrices in a manner that is simultaneously efficient and yet allows programming with high-level constructs. The fact that there is still no ...

    Real-Time C++ Efficient Object-Oriented and Template Microcontroller Programming

    标题《Real-Time C++:高效面向对象和模板微控制器编程》和副标题“第三版”,说明本书是关于C++编程语言在微控制器编程领域的应用,特别是面向对象编程(Object-Oriented Programming, OOP)和模板(template)编程。...

    Rheolef环境下基于C++的间断有限元编程方法

    Pierre Saramito所著的《Efficient C++ Finite Element Computing with Rheolef: Volume 2: Discontinuous Galerkin Methods》一书提供了关于在Rheolef环境下利用C++编写间断有限元程序的详细指南,书中不仅介绍了...

    高斯—约当消去法解线性方程组(C++)

    高斯—约当消去法解线性...高斯—约当消去法是一种efficient且广泛应用的线性方程组解法,具有良好的 numerical stability和计算速度。但是,该方法也存在一些缺点,需要选择合适的主元,且消元操作的计算复杂度较高。

    c&c++程序设计经典书籍

    3. 《Effective C++》(第二版)与《More Effective C++》:这两本书由Scott Meyers撰写,提供了50条具体的建议,帮助C++程序员写出更高效、更清晰、更少错误的代码。Meyers的"Effective"系列书籍是提高C++编程技巧...

    操作系统(内存管理)

    在大部分系统语言中,比如 C 和 C++,您必须进行内存管理。本文将介绍手工的、半手工的以及自动的内存管理实践的基本概念。 追溯到在 Apple II 上进行汇编语言编程的时代,那时内存管理还不是个大问题。您实际上在...

    easyloggingpp:单头C ++日志记录库。 它功能强大,可扩展,重量轻,性能快,线程和类型安全,并且具有许多内置功能。 它提供了以您自己的自定义格式编写日志的功能。 它还为记录您的类,第三方库,STL和第三方容器等提供支持

    10. **第三方库和容器支持**:可以直接记录STL容器(如vector、map等)以及第三方库中的对象,无需手动序列化。 使用`easyloggingpp`,开发者可以在代码中轻松添加日志记录,例如: ```cpp #include "easylogging+...

    Python 中文手册

    该站点上也提供了Python 的一些第三方模块,程序,工具,以及 附加的文档。 The Python interpreter is easily extended with new functions and data types implemented in C or C++ (or other languages callable ...

    树莓派编译而成的tensorflow动态库

    2. **THIRD_PARTY_TF_C_LICENSES**: 这个文件很可能包含了TensorFlow所依赖的第三方库的许可信息。TensorFlow作为一个大型项目,会引用许多其他的开源库,这些库也有各自的许可条款。 3. **lib**: 这个目录下应该是...

    Library for Adaptive Wavelet Application-开源

    同时,它使用了FLENS(Flexible Library for Efficient Numerical Solutions)作为构建块,FLENS是一个轻量级的C++库,用于支持数值计算,尤其是线性代数运算。 **小波分析** 小波分析是一种数学工具,结合了频率...

    ThunderGP:FPGA上基于HLS的图处理框架

    ThunderGP在获得第三名,在79个团队中排名前9。 ThunderGP被接受为 。 阅读 。 包含ThunderGP。 在。 请参阅, , 。介绍ThunderGP使数据科学家能够在不影响可编程性的前提下享受基于FPGA的图形处理的性能。 据我们...

Global site tag (gtag.js) - Google Analytics