使用 Delphi 2010,假设我有一个这样声明的类:
TMyList = TList<TMyObject>
对于这个列表,Delphi 为我们提供了一个枚举器,所以我们可以这样写:
var L:TMyList;
E:TMyObject;
begin
for E in L do ;
end;
麻烦的是,我想写这个:
var L:TMyList;
E:TMyObject;
begin
for E in L.GetEnumerator('123') do ;
end;
也就是说,我希望能够使用一些标准为同一个列表提供多个枚举器。不幸的是,实现for X in Z
需要一个函数Z.GetEnumerator
,没有参数,返回给定的枚举数!为了避免这个问题,我定义了一个实现“GetEnumerator”函数的接口,然后我实现了一个实现该接口的类,最后我在 TMyList 上编写了一个返回接口的函数!而且我正在返回一个界面,因为我不想为手动释放非常简单的类而烦恼......无论如何,这需要大量的输入。这是它的样子:
TMyList = class(TList<TMyObject>)
protected
// Simple enumerator; Gets access to the "root" list
TSimpleEnumerator = class
protected
public
constructor Create(aList:TList<TMyObject>; FilterValue:Integer);
function MoveNext:Boolean; // This is where filtering happens
property Current:TTipElement;
end;
// Interface that will create the TSimpleEnumerator. Want this
// to be an interface so it will free itself.
ISimpleEnumeratorFactory = interface
function GetEnumerator:TSimpleEnumerator;
end;
// Class that implements the ISimpleEnumeratorFactory
TSimpleEnumeratorFactory = class(TInterfacedObject, ISimpleEnumeratorFactory)
function GetEnumerator:TSimpleEnumerator;
end;
public
function FilteredEnum(X:Integer):ISimpleEnumeratorFactory;
end;
使用它我终于可以写:
var L:TMyList;
E:TMyObject;
begin
for E in L.FilteredEnum(7) do ;
end;
你知道更好的方法吗?也许 Delphi 确实支持直接使用参数调用 GetEnumerator 的方式?
后期编辑:
我决定使用 Robert Love 的想法,即使用匿名方法实现枚举器并使用 gabr 的“记录”工厂来保存另一个类。这允许我创建一个全新的枚举器,包含代码,只在函数中使用几行代码,不需要新的类声明。
这是我的通用枚举器在库单元中的声明方式:
TEnumGenericMoveNext<T> = reference to function: Boolean;
TEnumGenericCurrent<T> = reference to function: T;
TEnumGenericAnonim<T> = class
protected
FEnumGenericMoveNext:TEnumGenericMoveNext<T>;
FEnumGenericCurrent:TEnumGenericCurrent<T>;
function GetCurrent:T;
public
constructor Create(EnumGenericMoveNext:TEnumGenericMoveNext<T>; EnumGenericCurrent:TEnumGenericCurrent<T>);
function MoveNext:Boolean;
property Current:T read GetCurrent;
end;
TGenericAnonEnumFactory<T> = record
public
FEnumGenericMoveNext:TEnumGenericMoveNext<T>;
FEnumGenericCurrent:TEnumGenericCurrent<T>;
constructor Create(EnumGenericMoveNext:TEnumGenericMoveNext<T>; EnumGenericCurrent:TEnumGenericCurrent<T>);
function GetEnumerator:TEnumGenericAnonim<T>;
end;
这是一种使用它的方法。在任何类上,我都可以添加这样的函数(并且我有意创建一个不使用 aList<T>
来展示这个概念的强大功能的枚举器):
type Form1 = class(TForm)
protected
function Numbers(From, To:Integer):TGenericAnonEnumFactory<Integer>;
end;
// This is all that's needed to implement an enumerator!
function Form1.Numbers(From, To:Integer):TGenericAnonEnumFactory<Integer>;
var Current:Integer;
begin
Current := From - 1;
Result := TGenericAnonEnumFactory<Integer>.Create(
// This is the MoveNext implementation
function :Boolean
begin
Inc(Current);
Result := Current <= To;
end
,
// This is the GetCurrent implementation
function :Integer
begin
Result := Current;
end
);
end;
下面是我如何使用这个新的枚举器:
procedure Form1.Button1Click(Sender: TObject);
var N:Integer;
begin
for N in Numbers(3,10) do
Memo1.Lines.Add(IntToStr(N));
end;