深入理解C++的对象模型

何为C++对象模型?

C++对象模型可以概括为以下2部分:

1. 语言中直接支持面向对象程序设计的部分

2. 对于各种支持的底层实现机制

引言

现在有一个Point类,声明如下:

class Point {
 public:
 Point(float xval);
 virtual ~Point();

 float x() const;
 static int PointCount();

 protected:
 virtual ostream& print(ostream &os) const;

 float _x;
 static int _point_count;
};

这个类在机器上是通过什么模型来表示的呢?下面就介绍三种不同的实现方式。

1. 简单对象模型

简单对象模型名副其实,十分简单。在简单对象模型中,一个 object是由一系列slots组成,每个slot相当于一个指针,指向一个membermemebers按照声明的顺序与slots一一对应,这里的members括data membersfunction members

如果将简单对象模型应用在Point Class上,结构图如下:

优点:十分简单,降低了编译器设计的复杂度。

缺点:空间和时间上的效率降低。由于所有member都对应一个slot指针,所以每个object在空间上额外多出:member's number 乘以指针大小的空间。同时由于访问object的每个member都需要一次slot的额外索引,所以在时间的效率也会降低。

2. 表格驱动对象模型

表格驱动对象模型将member datamember function分别映射成两个表格member data tablefunction member table,而object本身只存储指向这两个表格的指针。 其中function member table是由一系列的slot组成,每个slot指向一个member function; member data table则直接存储的member data本身。如果将表格驱动对象模型应用在Point Class上,结构图如下:

优点:采用两层索引机制,对object变化提供比较好的弹性,在objectnonstatic data member有所改变时,而应用程序代码没有改变,这时是不需要重新编译的。

缺点:空间和时间上的效率降低,具体原因可以参考简单对象模型的缺点分析。

3. C++ 对象模型

Stroustrup 早期设计的C++对象模型是从简单对象模型改进而来的,并对内存空间和存取时间进行了优化。主要是将nonstatic data members存储在每一个object中,而static data members以及所有的function members被独立存储在所有object之外。

对虚函数的支持主要通过以下几点完成的:

所有包含虚函数或者继承自有虚函数基类的class都会有一个virtual table,该虚函数表存储着一堆指向该类所包含的虚函数的指针。

每个class所关联的type_info object也是由virtual table存储的,一般会存在该表格的首个slottype_info用于支持runtime type identification (RTTI)。

如果将C++对象模型应用在Point Class上,结构图如下:

优点:空间和存取效率高,所有static data members以及所有的function members被独立存储在所有object之外,可以减少每个object的大小,而nonstatic data members存储在每一个object中,又提升了存取效率。

缺点:如果应用程序的代码未曾更改,但所用到的classnonstatic data members有所更改,那么那些代码仍然需要全部重新编译,而前面的表格驱动模型在这方面提供了较大的弹性,因为他多提供了一层间接性,当然是付出了时间和空间上的代价。

在加上继承情况下的对象模型

C++支持单继承、多继承、虚继承,下面来看下base class实体在derived class中是如何被构建的。

简单对象模型中可以通过derived class object中的一个slot来存储base class subobject的地址,这样就可以通过该slot来访问base class的成员。这种实现方式的主要缺点是:因为间接性的存储而导致空间和存取时间上存在额外负担;优点是:derived class的结构不会因为base class的改变而改变。

表格驱动对象模型中可以利用一个类似base class table的表格来存储所有基类的信息。该表格中存储一系列slot,每个slot存储一个base class的地址。这种实现方式的缺点是:因为间接性的存储而导致空间和存取时间上存在额外负担;优点是:一是所有继承的class都有一致的表现形式(包含一个base table指针,指向基类表)与基类的大小和数目没有关系,二是base class table增加了子类的扩展性,当基类发生改变时,可以通过扩展、缩小或者更改base class table来进行调整。

以上两种实现方式都存在一个重要的问题,就是由于间接性而导致的空间和时间上的额外负担,并且该间接性的级数会随着继承的深度而增加。

C++ 最初采用的继承模型并不采用任何间接性,所有基类的数据直接存储在子类当中,这样在存储结构和访问效率上是最高效的。当然也有缺点:当base class members有任何改变,用到此base class或者derived class的对象必须重新编译。在C++ 2.0引入了virtual base class,需要一些间接性的方式来支持该特性,一般会导入一个virtual base class table或者扩展已有的virtual table

总结

以上就是深入研究C++的对象模型的全部内容,希望本文的内容对大家有所帮助。

(0)

相关推荐

  • 细品javascript 寻址,闭包,对象模型和相关问题

    正是因为JS是动态语言,所以JS的寻址是现场寻址,而非像C一样,编译后确定.此外,JS引入了this指针,这是一个很麻烦的东西,因为它"隐式"作为一个参数传到函数里面.我们先看"作用域链"话题中的例子: var testvar = 'window属性'; var o1 = {testvar:'1', fun:function(){alert('o1: '+this.testvar);}}; var o2 = {testvar:'2', fun:function(){

  • JavaScript Window浏览器对象模型方法与属性汇总

    Window 对象 所有浏览器都支持 window 对象.它表示浏览器窗口. 所有 JavaScript 全局对象.函数以及变量均自动成为 window 对象的成员. 全局变量是 window 对象的属性. 全局函数是 window 对象的方法. 1. open方法 语法格式: window.open(URL,窗口名称,窗口风格) 功能:打开一个新的窗口,并在窗口中装载指定URL地址的网页. 说明: open方法用于打开一个新的浏览器窗口,并在新窗口中装入一个指定的URL地址: open方法在打

  • C++ 多重继承和虚拟继承对象模型、效率分析

    一.多态 C++多态通过继承和动态绑定实现.继承是一种代码或者功能的传承共享,从语言的角度它是外在的.形式上的,极易理解.而动态绑定则是从语言的底层实现保证了多态的发生--在运行期根据基类指针或者引用指向的真实对象类型确定调用的虚函数功能!通过带有虚函数的单一继承我们可以清楚的理解继承的概念.对象模型的分布机制以及动态绑定的发生,即可以完全彻底地理解多态的思想.为了支持多态,语言实现必须在时间和空间上付出额外的代价(毕竟没有免费的晚餐,更何况编译器是毫无感情): 1.类实现时增加了virtual

  • JavaScript对象模型-执行模型

    简单数值类型: 有Undefined, Null, Boolean, Number和String.注意,描述中的英文单词在这里仅指数据类型的名称,并不特指JS的全局对象N an, Boolean, Number, String等,它们在概念上的区别是比较大的. 对象: 一个无序属性的集合,这些属性的值为简单数值类型.对象或者函数.同上,这里的对象并不特指全局对象Object. 函数: 函数是对象的一种,实现上内部属性[[Class]]值为"Function",表明它是函数类型,除了对象

  • SharePoint 客户端对象模型 (一) ECMA Script

    所谓的客户端对象模型就是调用了背后的WCF服务来提供数据,为了减轻数据的访问量数据包使用JSON,我们还可以看到对象模型的设计也加入了诸多对于减轻数据访问量的考量.技术上没有什么新意,你要愿意,在SharePoint2007里面也可以实现类似的功能,当然在使用上方便了我们不少  三种使用客户端模型的.NET托管.ECMA脚本,SilverLightClient. 本文讲阐述如何使用.NET托管代码来访问SharePoint对象模型. ECMAScript Client OM需要注意的几个点 EC

  • 理解Javascript_10_对象模型

    对象模型 红色虚线表示隐式Prototype链. 这张对象模型图中包含了太多东西,不少地方需要仔细体会,可以写些测试代码进行验证.彻底理解了这张图,对JavaScript语言的了解也就差不多了.下面是一些补充说明: 1. 图中有好几个地方提到build-in Function constructor,这是同一个对象,可以测试验证: 复制代码 代码如下: //Passed in FF2.0, IE7, Opera9.25, Safari3.0.4 Function==Function.constr

  • JavaScript 对象模型 执行模型

    数据类型 基本数据类型 基本数据类型是JS语言最底层的实现. 简单数值类型: 有Undefined, Null, Boolean, Number和String.注意,描述中的英文单词在这里仅指数据类型的名称,并不特指JS的全局对象N an, Boolean, Number, String等,它们在概念上的区别是比较大的. 对象: 一个无序属性的集合,这些属性的值为简单数值类型.对象或者函数.同上,这里的对象并不特指全局对象Object. 函数: 函数是对象的一种,实现上内部属性[[Class]]

  • 什么是DOM(Document Object Model)文档对象模型

    D:document 文档 浏览器加载的页面 DOM O:object 对象 页面及页面中的任何元素都是对象 M:module 模型 页面中的元素的组织形式 DOM被W3C组织设计为一种平台无关.语言无关的API,程序或脚本通过其动态访问.修改文档的内容.样式.结构. DOM是web 浏览器的运行规范,javascript借助DOM成就了其web标准语言的地位,在web领域实现了所谓"一次编写到处运行"的目标. 文档对象模型(Document Object Model,DOM)是一种用

  • JavaScript 浏览器对象模型BOM使用介绍

    BOM也叫做浏览器对象模型,它提供了很多对象,用于访问浏览器的功能;这些功能与任何网页内容无关; BOM缺少规范,每个浏览器提供商都按照自己的想法去扩展它,那么浏览器共有对象就成了事实的标准; 一 window对象 // BOM的核心对象是window,它表示浏览器的一个实例; // window对象处于JavaScript结构的最顶层; // 对于每个打开的窗口,系统都会自动为其定义window对象; // window对象同时扮演着ECMAScript中Global对象的角色,因此所有在全局

  • 简述JavaScript对传统文档对象模型的支持

    这是将其在JavaScript语言早期版本中引入的模型.大家都被所有浏览器都支持,但只允许访问文件的某些关键部分,如表单,表单元素和图像. 该模型提供了若干个只读属性,如标题,URL和上次更改提供关于文档整体的信息.除了有由该模型可用于设置和获取文档的属性值提供各种方法. 文档属性在传统DOM: 下面是文档属性,可以使用传统DOM访问列表:  文档方法在传统DOM: 这里是由传统DOM支持的方法列表:  例子: 我们可以找到任何HTML元素,使用HTML DOM任何HTML文档.例如,如果一个网

随机推荐