在前面文章《C++对象模型》可以知道,class 中成员函数的类型有:nonstatic member function、static member function、virtual member function;不同类型的成员函数的调用会有不同的表现,以下对每种类型的成员函数进行简单的分析。
为了使 nonstatic member function 能够与非成员函数具有相同的效力,C++ 编译器内部将 nonstatic member function 转换为非成员函数,转换的步骤以下:
例如:
class example{
public:
void func();
};
/* 其中成员函数 func() 会被编译器内部转换为 void func(example *this)*/
Static member function 和 nonstatic member function 1样会被编译器内部转换为非成员函数,区分是转换后不存在 this 指针,因此和全局函数1样,可以做 callback 函数。Static member function 有以下的特性:
在独立的 class 中 virtual member function 的调用有两种方式:指针调用 和 对象调用。例如:
class A {
virtual void func();
};
A a;
A *pA;
a.func();
pA->func();
用指针调用pA->func()
在编译器内部被转换为( * pA->vptr[1])(pA)
;
用对象调用 a.func()
在内部被当作和非虚拟成员函数1样处理,即转换为A::func(&a)
;所以 virtual 可以inline,在用对象调用的情况下inline 被展开。
具有 virtual member function 的 class,编译器对其产生1个 vtable, 把指向虚函数地址的指针放在该表格中;继承含有 virtual member function 类的派生类具有以下的特性:
像单继承1样,用基类的指针或援用只能访问基类中定义(或继承)的成员,不能访问派生类中引入的成员。当1个类继承于多个基类的时候,那些基类之间没有隐含的关系,不允许使用1个基类的指针访问其他基类的成员。
在多重继承的 virtual function 机制中,其复杂度围绕在第2个及后继的 base class 中,和在履行期间调剂 this 指针;1般规则经过指向第2或后继 base class 的指针或援用来调用 derived class virtual function。
在虚继承下,对给定虚基类,不管该类在派生层次中作为虚基类出现多少次,只继承1个同享的基类子对象。同享的基类子对象称为虚基类。
class A {
void func();
virtual void x();
};
对非虚函数void func()
,取它的指针得到的是函数的实际地址(即绑定在某个 class object 的地址),如:void (A::*pfunc) () = &A::func;
;
对虚函数virtual void x()
,取它的地址得到的是x
在 class A
的 vtable 中的索引值,如:viod (A::*pfunc) () = &A::x
;
使用 class object 或指向class object 的指针来调用,即用a.*pfunc()
或 pa->*pfunc()
调用;
当我们在 class 中定义 inline function 时,编译器会根据函数的复杂程度决定是不是真正定义为 inline 类型,对不能真正定义为 inline function 的要求,编译器可能把它们处理为 static member function;真实的 inline function 的扩大操作是在调用的那1点上,这会带来参数的求值操作和临时性对象的管理;