[原] Inside C++ Object model 笔记(3)

发布时间:2011-12-6 17:06
分类名称:C++


The Semantics of Function

Member 的各种调用方式

Nonstatic Member Functions

obj.magnitude(); 转换为:magnitude_7Point3dFv( register Point3d * const this);

this用来函数内部使用。关于函数名字变形和处理问题,见本书内容。不做详细说明。

Virtual Member Functions

ptr->normalize(); 转化为:( *(ptr->_vptr[1]) )(ptr);

看看,这就是所谓的"运行时"绑定。 _vptr[1],也就是虚函数表中的第一个函数地址,会随着程序代码的不同情况,有所变化。例如,ptr是一个Derived Class,那么这个位置,很有可能被Derived Class 替换掉。那么这个调用就会去调用子类的虚函数,而不是基类的。

一个class只会有一个virtual table,每一table内含有其对应的class object中所有的active virtual functions的实际地址,这些函数包括:

  1. class所定义的functions,这些functions可能会改写(overriding)一个已经存在的base class functions。
  2. 继承自base class的functions(没有被derived class改写)。
  3. 一个pure_virtual_called() 函数实体,扮演空间保卫和异常处理。


单一继承下的Virtual functions

class Point
{
public:
    virtual ~Point();
    virtual Point &mult(float) = 0;
    
    // ... ...
    float x() const { return _x;}
    virtual float y() const {return 0;}
    virtual float z() const {return 0;}
    // ... ...

protected:
    Point(float x = 0.0);
    float _x;
};

class Point2d : public Point
{
public:
    Point2d(float x = 0.0, float y = 0.0) : Point(x), _y(y) { }
    ~Point2d();

    // override base class virtual functions
    Point2d& mult(float);
    float y() const {return _y};
protected:
    float _y;
};

class Point3d : public Point2d
{
public:
    Point3d(float x = 0.0, float y = 0.0, float z = 0.0)
        : Point2d(x, y), _z(z) { }
    ~Point3d();

    // override base class virtual functions
    Point3d& mult(float);
    float z() const {return _z};
protected:
    float _z;
};


要说明的是,以上三张表,在运行的时候实际上都是存在的。子类只是拷贝了父类的表格,然后进行改写。而不是直接拿来父类的表格在上面写写画画。为什么,待会儿我会摘取《ATL Under the hood》中的一节来说明。

多继承下的Virtual Functions

多重继承的问题在于Dervied class和第二个和其后继的base classes的this指针调节上。

class Base1
{
public:
    Base1();
    virtual ~Base1();
    virtual void speakClearly();
    virtual Base1 *clone() const;
protected:
    float data_Base1;
};

class Base2
{
public:
    Base2();
    virtual ~Base2();
    virtual void mumble();
    virtual Base2 *clone() const;
protected:
    float data_Base2;
};

class Derived : public Base1, public Base2
{
public:
    Derived();
    virtual ~Derived();
    virtual Derived *clone() const;
protected:
    float data_Derived;
};

Base2 *pBase2 = new Derived; ==>>转换为:

Derived *pTemp = new Derived;
Base2* pBase2 = pTemp ? (Base2 *)((char *)pTemp + sizeof(Base1)) : 0;

当要执行: delete pBase2的时候,需要再次调整回,以便指到对象的起始点。然而,这个offset在编译时刻无法确定。所以调整需要在执行期间来进行。比较有效率的做法是使用Thunk技术(就是插入一小段汇编)。看看VS2010的Derived布局:

+---
| +--- (base class Base1)
0 | | {vfptr}
4 | | data_Base1
| +---
| +--- (base class Base2)
8 | | {vfptr}
12 | | data_Base2
| +---
16 | data_Derived
+---
Derived::$vftable@Base1@:
| &Derived_meta
| 0
0 | &Derived::{dtor}
1 | &Base1::speakClearly
2 | &Derived::clone
Derived::$vftable@Base2@:
| -8
0 | &thunk: this-=8; goto Derived::{dtor} ;看到没,这里有几个thunk字样,后面调整this,然后调用子类函数。
1 | &Base2::mumble
2 | &thunk: this-=8; goto Base2* Derived::clone
3 | &thunk: this-=8; goto Derived* Derived::clone
Derived::{dtor} this adjustor: 0
Derived::clone this adjustor: 0
Derived::__delDtor this adjustor: 0
Derived::__vecDelDtor this adjustor: 0


虚拟继承下的Virtual Functions

class Point2d
{
public:
    Point2d(float x = 0.0, float y = 0.0){ }
    virtual ~Point2d();

    // override base class virtual functions
    virtual void mumble();
    virtual float z();
protected:
    float _x, _y;
};

class Point3d : virtual public Point2d
{
public:
    Point3d(float x = 0.0, float y = 0.0, float z = 0.0)
        : Point2d(x, y), _z(z) { }
    ~Point3d();

    // override base class virtual functions
    Point3d& mult(float);
    float z() const {return _z;}
protected:
    float _z;
};

微软编译器布局如下:

class Point3d size(20):
 +---
 0 | {vbptr}
 4 | _z
 +---
 +--- (virtual base Point2d)
 8 | {vfptr}
12 | _x
16 | _y
 +---
Point3d::$vbtable@:
 0 | 0
 1 | 8 (Point3dd(Point3d+0)Point2d)
Point3d::$vftable@:
 | -8
 0 | &Point3d::{dtor}
 1 | &Point2d::mumble
 2 | &Point2d::z
Point3d::{dtor} this adjustor: 8
Point3d::__delDtor this adjustor: 8
Point3d::__vecDelDtor this adjustor: 8
vbi:    class  offset o.vbptr  o.vbte fVtorDisp
         Point2d       8       0       4 0

不要在virtual base clas中声明nonstatic data member。为什么?你可以自己试试。


Static Member Functions

没有啥好说的,Static Member Functions就和一般的普通函数(非成员函数)一样,该怎么调用,就怎么调用。这种函数是憋屈了一段时间才被推出的,以前有种奇怪的调用方法:((Point3d *)0)->object_count(); 在引入static member functions之前,C++语言要求所有的member functions都必须由该class的object调用,所有就出现了上述调用方法。