我有一个类实例和对其的两个引用。有一个对象引用(aMII : TMyInterfaceImpl)和一个接口引用(iMI : IMyInterface)。
当我使用对象引用将其方法之一分配给方法类型变量(methodRef : TMyMethodRef)时,它可以正常工作。

但是,如果我想在相同情况下使用接口引用,则编译器将引发错误:


  [dcc32错误] Unit1.pas(66):E2010不兼容的类型:“ TMyMethodRef”和“过程,无类型的指针或无类型的参数”


最小的例子:

type
  TMyMethodRef = procedure of object;
  //TMyMethodRef = procedure of interface; // Invalid definition, just a desperate attempt

  IMyInterface = interface
    ['{BEA60A2B-C20F-4E02-A938-65DD4332ADB0}']
    procedure foo;
  end;

  TMyInterfaceImpl = class ( TInterfacedObject, IMyInterface )
    procedure foo;
  end;

procedure TMyInterfaceImpl.foo;
begin
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  aMII : TMyInterfaceImpl;
  iMI : IMyInterface;
  methodRef : TMyMethodRef;
begin
  aMII := TMyInterfaceImpl.Create;
  iMI := aMII;
  methodRef := aMII.foo; // It compiles
  methodRef := iMI.foo; // It does not compile

end;


在我的原始资料中,我必须使用接口(由于抽象的工厂设计模式:function TMyFactory.createMyInterface : IMyInterface,所以没有对象引用)。

而且我必须将其方法之一用作某些事件的回调函数(作为参数传递给setter方法:procedure TMyInterfaceImpl2.setOnEvent( onEvent_ : TMyMethodRef )

我该怎么做?如何将接口引用的方法之一分配给方法类型变量?

我知道有一种解决方案可以实现另一个接口来访问对象引用,但是它并不安全(我无法始终确保类同时实现两个接口),并且我不希望这种解决方案:

type
  TMyMethodRef = procedure of object;

  IMyInterface = interface
    ['{BEA60A2B-C20F-4E02-A938-65DD4332ADB0}']
    procedure foo;
  end;

  TMyInterfaceImpl = class;

  IInstanceAccessor<T> = interface
    ['{61E36FF2-A9D3-4B46-BAE8-217CBE7A1F18}']
    function getInstance : T;
  end;

  TMyInterfaceImpl = class ( TInterfacedObject, IMyInterface, IInstanceAccessor<TMyInterfaceImpl> )
    procedure foo;
    function getInstance : TMyInterfaceImpl;
  end;

procedure TMyInterfaceImpl.foo;
begin
end;

function TMyInterfaceImpl.getInstance : TMyInterfaceImpl;
begin
  result := self;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  iMI : IMyInterface;
  aMII : TMyInterfaceImpl;
  methodRef : TMyMethodRef;
begin
  iMI := TMyInterfaceImpl.Create;
  aMII := ( iMI as IInstanceAccessor<TMyInterfaceImpl> ).getInstance;
  methodRef := aMII.foo;
end;

最佳答案

你不能这样做。

您可以做的就是使用这种方法声明适配器类,然后将该方法用作回调。

type
  TMyInterfaceAdapter = class
  private
    FIntf: IMyInterface;
  public
    procedure Foo;
    property Intf: IMyInterface read FIntf write FIntf;
  end;

procedure TMyInterface.Foo;
begin
  Assert(Assigned(FIntf), 'no interface assigned');
  FIntf.Foo;
end;

10-08 04:48