如何将TObjectList用于任意类类型?

时间:2018-01-11 15:31:35

标签: delphi generics inheritance delphi-10.1-berlin

我对Delphi中的泛型仍然有些模糊,但是已经广泛使用TObjectList<>。现在我的情况是我有一个带有这样一个私有字段的基类,但需要为任意类创建,也从另一个基类继承。

为了澄清,我有两个基类:

type
  TItem = class;
  TItems = class;

  TItemClass = class of TItem;

  TItem = class(TPersistent)
  private
    FSomeStuffForAllIneritedClasses: TSomeStuff;
  end;

  TItems = class(TPersistent)
  private
    FItems: TObjectList<TItem>;
    FItemClass: TItemClass;
  public
    constructor Create(AItemClass: TItemClass);
    destructor Destroy; override;
    function Add: TItem;
    ...
  end;

然后将这对类进一步继承到更具体的类中。我希望对所有这些对象列表共享,而每个对象列表在内部实际上都是一个不同的类型。

type
  TSomeItem = class(TItem)
  private
    FSomeOtherStuff: TSomeOtherStuff;
    ...
  end;

  TSomeItems = class(TItems)
  public
    function Add: TSomeItem; //Calls inherited, similar to a TCollection
    procedure DoSomethingOnlyThisClassShouldDo;
    ...
  end;

现在问题在于创建实际的对象列表。我试着这样做:

constructor TItems.Create(AItemClass: TItemClass);
begin
  inherited Create;
  FItemClass:= AItemClass;
  FItems:= TObjectList<AItemClass>.Create(True);
end;

然而,代码洞察力抱怨这个:

  

未声明的标识符AItemClass

更多的是,编译器还有一个不同的抱怨:

  

未声明的标识符TObjectList

其中,我确实在本单元中使用了System.Generics.Collections

我在这里做错了什么,我应该怎么做呢?

2 个答案:

答案 0 :(得分:6)

TItems通用:

TItems<T: TItem, constructor> = class(TPersistent)
private
  FItems: TObjectList<T>;
public
  constructor Create;
  destructor Destroy; override;
  function Add: T;
  ...
end;

constructor TItems.Create;
begin
  inherited Create;
  FItems:= TObjectList<T>.Create(True);
end;

function TItems<T>.Add: T;
begin
  Result := T.Create;
  FItems.Add(Result);
end;

如果继承,只需输入正确的通用参数:

TSomeItems = class(TItems<TSomeItem>)
public
  procedure DoSomethingOnlyThisClassShouldDo;
  ...
end;

答案 1 :(得分:2)

TObjectList并不意味着以这种方式使用。它最初被定义为TObjectList<TItem>这一事实意味着它也希望您以这种方式创建它。它需要使用您打算创建它的精确类来定义。

相反,只需使用TItem创建它,然后每当您创建一个应该添加到此列表的新项目时,然后您使用类类型创建它。每当您需要访问此列表中的项目时,只需动态投射它们。

例如......

Result:= FItemClass.Create;
FItems.Add(Result);

...可以是Add函数的内容。