假设我有这个定义:
TMyClass1 = class
end;
TMyClass2 = class
end;
IModel<T : class> = interface
['{E8262D6C-DCAB-46AC-822E-EC369CF734F8}']
function List() : TObjectList<T>;
end;
IPresenter<T : class> = interface
['{98FB7751-D75A-4C51-B55A-0E5FE68BE213}']
function Retrieve() : TObjectList<T>;
end;
IView<T : class> = interface
['{59384CD6-30D6-4BD8-AB3D-7FCF4D1A8618}']
procedure AssignPresenter(APresenter : IPresenter<T>);
end;
TModel<T : class> = class(TInterfacedObject, IModel<T>)
public
function List() : TObjectList<T>; virtual; abstract;
end;
TPresenter<T : class> = class(TInterfacedObject, IPresenter<T>)
strict private
{ Private declarations }
FModel : IModel<T>;
FView : IView<T>;
public
constructor Create(AView : IView<T>);
function Retrieve() : TObjectList<T>; virtual; abstract;
end;
TModelClass1 = class(TModel<TMyClass1>);
TPresenterClass1 = class(TPresenter<TMyClass1>);
TModelClass2 = class(TModel<TMyClass2>);
TPresenterClass2 = class(TPresenter<TMyClass2>);
我有这个表单来实现我定义的一些东西:
TForm1 = class(TForm, IView<TMyClass1>, IView<TMyClass2>)
procedure FormCreate(Sender: TObject);
private
{ Private declarations }
FPresenter1 : IPresenter<TMyClass1>;
FPresenter2 : IPresenter<TMyClass2>;
procedure AssignPresenter(APresenter : IPresenter<TMyClass1>); overload;
procedure AssignPresenter(APresenter : IPresenter<TMyClass2>); overload;
end;
procedure TForm1.FormCreate(Sender: TObject);
begin
TPresenterClass1.Create((Self as IView<TMyClass1>));
TPresenterClass2.Create((Self as IView<TMyClass2>));
end;
procedure TForm1.AssignPresenter(APresenter: IPresenter<TMyClass1>);
begin
Self.FPresenter1 := APresenter;
end;
procedure TForm1.AssignPresenter(APresenter: IPresenter<TMyClass2>);
begin
Self.FPresenter2 := APresenter;
end;
所以这里的问题是delphi无法找出要调用的方法,在这个例子中,在这两种情况下只调用了AssignPresenter(APresenter: IPresenter<TMyClass2>)
,所以可能我在这里遗漏了一些东西,但我可以&#39 ;弄清楚atm。
答案 0 :(得分:3)
这可能是重复但我现在找不到它。
问题是接口的as
运算符与泛型不兼容。 as
运算符依赖于接口GUID。通过查询具有匹配GUID的接口找到该接口。并且GUID在通用实例化方面并不适合。
现在让我们来看看你的代码。
TPresenterClass1.Create((Self as IView<TMyClass1>));
TPresenterClass2.Create((Self as IView<TMyClass2>));
问题是IView<TMyClass1>
和IView<TMyClass2>
具有相同的GUID:
type
IView<T : class> = interface
['{59384CD6-30D6-4BD8-AB3D-7FCF4D1A8618}']
procedure AssignPresenter(APresenter : IPresenter<T>);
end;
因此,IView<TMyClass1>
和IView<TMyClass2>
共享相同的GUID,当您使用as
进行查询时,无论您是否要求IView<TMyClass1>
,都会返回相同的界面}或IView<TMyClass2>
。
因此,这里的底线是,只要对象使用不同的as
实现ISomeInterface<T>
两次,T
就会在通用接口旁边呈现无效。
Embarcadero确实应该以支持泛型的方式实现as
。我不会屏住呼吸。
您需要找到一种不同的方法来解决您的问题。
答案 1 :(得分:2)
当你拥有实现接口的对象时,不需要使用as
- 只需写:
procedure TForm1.FormCreate(Sender: TObject);
begin
TPresenterClass1.Create(Self);
TPresenterClass2.Create(Self);
end;
编译器正确地计算出来。当您使用as
时,它会进行Supports
调用,但由于David已解释的原因而失败。
答案 2 :(得分:0)
作为David Heffernan回答的补充。
解决问题的一种方法是为您使用的所有通用声明提供唯一的GUID:
IViewMyclass1 = interface(IView<TMyClass1>)
['{1A0F941F-BAB1-4723-A6C1-27036DF5D344}']
end;
IViewMyclass2 = interface(IView<TMyClass2>)
['{0C61A23A-DC50-43B0-97C9-8B0013DDC193}']
end;
重新定义视图的声明。
TForm4 = class(TForm, IViewMyclass1, IViewMyclass2)
procedure TForm1.FormCreate(Sender: TObject);
begin
TPresenterClass1.Create((Self as IViewMyclass1));
TPresenterClass2.Create((Self as IViewMyclass2));
end;
然后调用正确的重载。
免责声明: