今天一个好友在群里问:: 在VCL库里,大多数构造函数是virtual的,这是为啥: 按理,在调用构造函数之前,对象都还没有,: 当然也应该没有vtbl,那么这样virtual有什么意义呢?: 所以C++里的构造函数没有virtual的。: 难道Delphi里不是酱紫地?: 那么vtbl是在什么时候被初始化的?
首先,为什么没有调用构造函数之前,调用virtual函数没有意义,而调用实函数就有意义呢?因为虚方法和实方法不同,调用实方法的时候,实际上是调用类的方法,在实例没有创建前,就可以调用的。而虚方法不同,虚方法依赖于实例,必须要实例的vmt表建立后才能调用。
那么回到主题,为什么vcl里很多构造函数是虚函数呢?
因为vcl的初始化实际上都是在newInstance里完成的。 create只是把最后的堆内存,变成可用的指针 在delphi源代码里,可以看到:class function TObject.NewInstance: TObject;begin Result := InitInstance(_GetMem(InstanceSize));end; class function TObject.InitInstance(Instance: Pointer): TObject;{$IFDEF PUREPASCAL}var IntfTable: PInterfaceTable; ClassPtr: TClass; I: Integer;begin FillChar(Instance^, InstanceSize, 0); PInteger(Instance)^ := Integer(Self); ClassPtr := Self; while ClassPtr <> nil do begin IntfTable := ClassPtr.GetInterfaceTable; if IntfTable <> nil then for I := 0 to IntfTable.EntryCount-1 do with IntfTable.Entries[I] do begin if VTable <> nil then PInteger(@PChar(Instance)[IOffset])^ := Integer(VTable); end; ClassPtr := ClassPtr.ClassParent; end; Result := Instance;end;{$ELSE}可以看出,最终的构造是在newInstance里完成。