考虑继承,a tip about 《Inside VCL》

类别:Delphi 点击:0 评论:0 推荐:
考虑继承,a tip about 《Inside VCL》

先行知识:OO/VCL

难度:★☆☆☆☆

 

最近终于有时间看一看在前一段时间买的书《Inside VCL》,在读书的时候难免会发现一些问题,也许有些是微不足道的,但自己认为是以前没有考虑过的东西我便把它写出来(这篇也不例外)。

李维老师在对象基本服务那一节中讲到了对象的释放服务,可以看到一个简单的TObject.DestroyDelphi却在背后为我们隐藏了那么复杂的操作,在书中的80页里写到了被析构函数调用的ClearupInstance,在其中有这样的实现:

while (ClassPtr <> nil) and (InitTable <> nil) do

begin

  _FinalizeRecord(Self, InitTable);

  ClassPtr := ClassPtr.ClassParent;

  if ClassPtr <> nil then

    InitTable := PPointer(Integer(ClassPtr) + vmtInitTable)^;

end;

内存的具体释放工作在交给_FinalizeRecord实现后出现了这样的语句:

ClassPtr := ClassPtr.ClassParen

看到这里我有些不明白了?在释放了当前的对象所拥有的空间后为什么还要遍历到父类呢?难道我们释放了我们对象的空间后还要释放父类的空间?那不根本不是我们所要求的意思吗?(后来才发现这是由我以前没有考虑过的一个问题而引起的根本错误!)。我们考虑下下面的两个类和相关的代码:

TBase = Class (TObject)

 Private

  Field:string;

End;

 

TFooBase = Class (TBase)

 Private

  FooField: string;

End;

var

 Base:TBase;

 FooBase:TfooBase

在释放FooBase时会最终调用到_WstrArray来释放其私有字段FooField所占的空间,那么释放这种特定类型所占的空间是否到此就结束了呢?当然不是,因为还有一个字段没有释放呢!那就是TBase的Field!为什么需要释放这个字段呢?我们并不释放Base啊,那是因为在FooBase中也存在这个字段,我当时就在问Field不是TBase所私有的吗?和TfooBase有什么关系?原因就在这里了,即使Field是TBase所定义的私有成员但它仍将被TfooBase所继承,只是不能访问罢了!(这个问题在很多大学的OO教材中都没有提到过,只是说私有成员不能被派生的类访问,而到底派生的类中有没有这个成员就没有清楚的说了,有些甚至描述为基类中的私有成员不能被派生类所继承。后来在问了我的几个同学这个问题后,我更确定了这一点大家都没有想过…)。写到这里就不难解释上的ClassPtr := ClassPtr.ClassParen语句了。我们再通过一个例子来证实一下:

按上面的两个类创建两个实例,分别用showmessage(inttostr(Base.InstanceSize));和showmessage(inttostr(FooBase.InstanceSize));来查看对象的大小,可以看到基类的对象大小为8(一个self指针一个字符串的指针),派生类对象为大小为12(一个self指针2个字符串的指针),写到这里这个问题就清楚了:

ClassPtr := ClassPtr.ClassParen只是为了通过基类找到基类的vmtInitTable,从而找到基类有多少个字段各是什么类型以便在派生类中释放。

看来在日常的编程中仍有一些细小的问题是我没有考虑过的,以后应该多留意一些这些细节,不过这个问题我想大多数人都应该知道的,只是刚学OO的初学者可能没有仔细考虑过吧。

本文地址:http://com.8s8s.com/it/it5121.htm