4

我面临从 Spring4D 框架容器解析的类型正确实例化对象的问题。

我有一堂课:

type
  TSurvey = class ( TInterfacedObject, ISurvey )

  private
        _id : Integer;
        _organization : IOrganization;

        function GetId () : Integer;
        procedure SetId ( const value : Integer );

        function GetOrganization () : IOrganization;
        procedure SetOrganization ( const value : IOrganization);

  public
        property Id : Integer read GetId write SetId;
        property Organization: IOrganization read GetOrganization write SetOrganization;
end;

...

initialization

  GlobalContainer.RegisterType<TSurvey>.Implements<ISurvey>.InjectField ( '_organization' );

...

我使用 GlobalContainer 来实例化一个对象:

survey := GlobalContainer.Resolve<ISurvey>;
survey.Organization.Id := 5;

一切都很好,并且运行良好。

现在我想为 TSurvey 创建一个后代类:

type
  TFieldSurvey = class ( TSurvey )
  ...
end;

问题是如何正确实例化 TFieldSurvey 类的对象?

如果我使用 Create(),那么我会得到一个异常:

 fieldSurvey := TFieldSurvey.Create ();
 fieldSurvey.Organization.Id := 5    <- exception is here

我是否必须在 TFieldSurvey 构造函数中显式调用组织字段的构造函数,还是有另一种方法?例如,使用 GlobalContainer?

提前致谢。

4

2 回答 2

4

注入仅在您通过容器创建对象时起作用,而不是直接调用对象上的构造函数。所以你需要注册TFieldSurvey然后GlobalContainer调用Resolve来获取你的对象。

登记:

GlobalContainer.RegisterType<TSurvey>.Implements<ISurvey>('SPRING_SURVEY').InjectField ( '_organization' );
GlobalContainer.RegisterType<TFieldSurvey>.Implements<ISurvey>('SPRING_FIELD_SURVEY').InjectField ( '_organization' );

然后获取实例:

GlobalContainer.Resolve<ISurvey>('SPRING_FIELD_SURVEY')

我添加了“SPRING_SURVEY”和“SPRING_FIELD_SURVEY”的名称,因为它们都实现了ISurvey,这使您可以选择所需的类实例,否则您最终会得到为该接口注册的最后一个实现。如果TFieldSurvey要实现自己的接口(例如),您可以取消名称,然后在需要时IFieldSurvey重新类型转换回。ISurvey

您也可以始终[Inject]在 _organization 字段上使用该属性,而不是使用 .InjectField (在添加Global.Container.Common到您的用途之后):

  TSurvey = class ( TInterfacedObject, ISurvey )
  private
    _id : Integer;
    [Inject]
    _organization : IOrganization;

    function GetId () : Integer;
    procedure SetId ( const value : Integer );

    function GetOrganization () : IOrganization;
    procedure SetOrganization ( const value : IOrganization);

  public
    property Id : Integer read GetId write SetId;
    property Organization: IOrganization read GetOrganization write SetOrganization;
  end;

您的注册将是:

  GlobalContainer.RegisterType<TSurvey>.Implements<ISurvey>('SPRING_SURVEY');
  GlobalContainer.RegisterType<TFieldSurvey>.Implements<ISurvey>('SPRING_FIELD_SURVEY');
于 2015-07-31T05:17:42.363 回答
2

您不应该以仅适用于 DI 容器的方式编写代码。

DI 容器是一种工具,您应该避免(直接或间接)依赖它。

这意味着您应该避免使用字段注入,因为这样的代码不能与纯 DI一起使用 - 请改用构造函数或属性注入。

同样从您发布的代码片段中,我可以闻到服务定位器反模式

如果您想创建调查,请使用调查工厂并将其注入您正在使用它的类中。DI 容器通常不用于创建值对象,并且您的调查类(尽管不必要地具有接口)看起来像一个。

在深入使用 DI 容器之前,我真的建议更多地了解 DI 的工作原理以及哪些技术可以让您使用 DI 编写干净的代码。然后才开始使用 DI 容器。反过来做只会导致错误,最终使代码更难维护。

于 2015-07-31T10:19:00.690 回答