现在准备一步步地模拟 TList 类, 建立一个自己的 TMyList.
首先, 这个类中应该包括前面提到的那个 Pointer 数组(TPointerList)的指针(PPointerList):
首先, 这个类中应该包括前面提到的那个 Pointer 数组(TPointerList)的指针(PPointerList):
TMyList = class(TObject) FList: PPointerList; end;
既然是一个列表, 应该有 Count 字段:
TMyList = class(TObject) FList: PPointerList; FCount: Integer; end;
TList 类还有一个 FCapacity 字段.
譬如当前列表中有 10 个元素时, 其实列表已经提前申请出更多位置, 这样就不必每次添加每次申请内存, 从而提高了效率.
这个 FCapacity 字段就表示已申请的可放置元素位置的总数.
TMyList = class(TObject) FList: PPointerList; FCount: Integer; FCapacity: Integer; end;
它们都应该是私有的:
TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; public end;
应该用属性操作它们:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs; type TForm1 = class(TForm) end; {--------------------------------} TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; public property Capacity: Integer; {光标放在属性上, 执行 Shift+Ctrl+C, 自动建立属性} property Count: Integer; property List: PPointerList; end; {--------------------------------} var Form1: TForm1; implementation {$R *.dfm} end.
自动建立后的属性:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs; type TForm1 = class(TForm) end; {--------------------------------} TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; procedure SetCapacity(const Value: Integer); procedure SetCount(const Value: Integer); procedure SetList(const Value: PPointerList); public property Capacity: Integer read FCapacity write SetCapacity; property Count: Integer read FCount write SetCount; property List: PPointerList read FList write SetList; end; {--------------------------------} var Form1: TForm1; implementation {$R *.dfm} { TMyList } procedure TMyList.SetCapacity(const Value: Integer); begin FCapacity := Value; end; procedure TMyList.SetCount(const Value: Integer); begin FCount := Value; end; procedure TMyList.SetList(const Value: PPointerList); begin FList := Value; end; end.
在 TList 中, SetCount 和 SetCapacity 方法都是放在 protected 区以供子类使用, 这里暂时不考虑继承的问题, 就让它们先在 private 区吧;
不过 List 属性应该是只读的, 它就是那个核心数组的指针, 对它的赋值应该通过更多其他的方法.
修改如下:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs; type TForm1 = class(TForm) end; {--------------------------------} TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; procedure SetCapacity(const Value: Integer); procedure SetCount(const Value: Integer); public property Capacity: Integer read FCapacity write SetCapacity; property Count: Integer read FCount write SetCount; property List: PPointerList read FList; end; {--------------------------------} var Form1: TForm1; implementation {$R *.dfm} { TMyList } procedure TMyList.SetCapacity(const Value: Integer); begin FCapacity := Value; end; procedure TMyList.SetCount(const Value: Integer); begin FCount := Value; end; end.
作为一个列表, 最起码应该有 Add、Delete、Clear 方法.
Add 方法的参数应该是一个指针(TList 就是存放指针的吗), 如果需要同时返回元素的位置, 应该定义成函数;
Delete 是删除指定位置上的元素, 参数是个位置号;
Clear 是清除列表, 对象释放(Destroy)时也应该执行此过程.
综上所述, 修改如下:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs; type TForm1 = class(TForm) end; {--------------------------------} TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; procedure SetCapacity(const Value: Integer); procedure SetCount(const Value: Integer); public destructor Destroy; override; {覆盖父类的 Destroy 方法} function Add(Item: Pointer): Integer; procedure Clear; {在 TList 中这是个虚方法, 以备子类覆盖} procedure Delete(Index: Integer); property Capacity: Integer read FCapacity write SetCapacity; property Count: Integer read FCount write SetCount; property List: PPointerList read FList; end; {--------------------------------} var Form1: TForm1; implementation {$R *.dfm} { TMyList } function TMyList.Add(Item: Pointer): Integer; begin end; procedure TMyList.Clear; begin end; procedure TMyList.Delete(Index: Integer); begin end; destructor TMyList.Destroy; begin Clear; {在类释放时同时清除列表} inherited; {在 TList 中没有这句, 大概是因为它的父类也没什么可继承的} end; procedure TMyList.SetCapacity(const Value: Integer); begin FCapacity := Value; end; procedure TMyList.SetCount(const Value: Integer); begin FCount := Value; end; end.
为了让代码更清晰, 还是把这个类相关的所有代码放到一个独立的单元吧(新建一个单元, 我们把它保存为: MyList.pas).
因为 MaxListSize、TPointerList、PPointerList 都是在 Classes 单元定义的, 为了省略 uses Classes, 在 MyList 单元中重新做了定义.
unit MyList; interface const MaxListSize = Maxint div 16; type PPointerList = ^TPointerList; TPointerList = array[0..MaxListSize - 1] of Pointer; TMyList = class(TObject) private FList: PPointerList; FCount: Integer; FCapacity: Integer; procedure SetCapacity(const Value: Integer); procedure SetCount(const Value: Integer); public destructor Destroy; override; function Add(Item: Pointer): Integer; procedure Clear; procedure Delete(Index: Integer); property Capacity: Integer read FCapacity write SetCapacity; property Count: Integer read FCount write SetCount; property List: PPointerList read FList; end; implementation { TMyList } function TMyList.Add(Item: Pointer): Integer; begin end; procedure TMyList.Clear; begin end; procedure TMyList.Delete(Index: Integer); begin end; destructor TMyList.Destroy; begin Clear; inherited; end; procedure TMyList.SetCapacity(const Value: Integer); begin end; procedure TMyList.SetCount(const Value: Integer); begin end; end.
至此, 类的轮廓已经出来, 该实现一下这些方法了.