Spring4d:如何"force"容器相信一个class实现一个接口

Spring4d: How to "force" the container to believe a class implements an interface

我正在使用 RemObjects DataAbstract 和 Spring4d。 RemObjects 生成一个 SchemaServer_Intf.pas 文件,其中包含其模式中存在的每种 table 的接口。它允许 "Strongly typed" 个数据集,允许使用

访问一个字段
(aDataSet as IMyDataSet).MyField := aValue

这是 DataAbstract 生成的其中一个接口的快照

IEntiteType = interface(IDAStronglyTypedDataTable)
  ['{96B82FF7-D087-403C-821A-0323034B4B99}']
    { Property getters and setters }
    function GetEntiteIdValue: String;
    procedure SetEntiteIdValue(const aValue: String);
    function GetEntiteIdIsNull: Boolean;
    procedure SetEntiteIdIsNull(const aValue: Boolean);
    function GetNameValue: WideString;
    procedure SetNameValue(const aValue: WideString);
    function GetNameIsNull: Boolean;
    procedure SetNameIsNull(const aValue: Boolean);
    function GetIsSystemValue: SmallInt;
    procedure SetIsSystemValue(const aValue: SmallInt);
    function GetIsSystemIsNull: Boolean;
    procedure SetIsSystemIsNull(const aValue: Boolean);


    { Properties }
    property EntiteId: String read GetEntiteIdValue write SetEntiteIdValue;
    property EntiteIdIsNull: Boolean read GetEntiteIdIsNull write SetEntiteIdIsNull;
    property Name: WideString read GetNameValue write SetNameValue;
    property NameIsNull: Boolean read GetNameIsNull write SetNameIsNull;
    property IsSystem: SmallInt read GetIsSystemValue write SetIsSystemValue;
    property IsSystemIsNull: Boolean read GetIsSystemIsNull write SetIsSystemIsNull;
  end;

不过,有一个问题。如果您像这样投射数据表:

aDataTable := IEntiteType(TDAMemDataTable.Create(nil));

您将遇到“不支持接口错误

但是,一旦你这样做:

aDataTable.LogicalName := 'EntiteType';
aDataTable.BusinessRulesId := MyBusinessRuleID;

你可以放心地写

aDataTable := IEntiteType(TDAMemDataTable.Create(nil));

而且你不会收到任何错误。

所以,有了Spring4d,我想到了在我的注册单元里这样写:

aContainer.RegisterType<TDAMemDataTable>.Implements<IEntiteType>.DelegateTo(
   function : TDAMemDataTable
   var aDataTable : TDAMemDataTable; 
   begin
      Result:= TDAMemDataTable.Create(nil);
      Result.LogicalName := 'EntiteType';
      Result.BusinessRulesId := MyBusinessRuleId;         
   end
) 

但是,Spring4d 抛出(有原因的)错误:

Exception 'first chance' à 2D5B68. Classe d'exception ERegistrationException avec un message 'Component type "uDAMemDataTable.TDAMemDataTable" incompatible with service type "SchemaClient_Intf.IEntiteType".'. Processus EntiteREM2.exe (3088) 

有没有办法覆盖这个检查?

好的,我找到了一种方法。其实超级简单:

  aContainer.RegisterType<IAddress>.DelegateTo(
    function : IAddress
    var aTable : TDAMemDataTable;
    begin
      aTable := TDAMemDataTable.Create(nil);
      aTable.LogicalName := nme_Address;
      aTable.BusinessRulesID := RID_Address;
      Result := aTable as IAddress;
    end
  );

此外,对于有兴趣以优雅的方式注册多张桌子的人:

aContainer.RegisterType<IAddress>.DelegateTo(TableConfigurator.GetTableDelegate<IAddress>(nme_Address, RID_Address));
// Registering other tables here...

只需使用此方法创建一些 "Helper" class 即可:

class function TableConfigurator.GetTableDelegate<T>(aLogicalName, aBusinessRulesId: string): TActivatorDelegate<T>;
begin
  Result := (function: T
    var
      aTable: TDAMemDataTable;
    begin
      aTable := TDAMemDataTable.Create(nil);
      aTable.LogicalName := aLogicalName;
      aTable.BusinessRulesID := aBusinessRulesId;
      Result := T(TValue.From(aTable).AsInterface);
    end);
end;