使用简单注入器的策略模式

时间:2017-07-11 17:50:47

标签: c# dependency-injection simple-injector

我有一个工作流程,我需要根据一些运行时数据使用特定的服务类。

例如。我有一个SendEmailCommand,其依赖关系为IEmailService,其中包含MandrillEmailServiceSmtpEmailService的两个实现。

SendEmailCommand将决定它在运行时使用的IEmailService实现。

每个电子邮件服务实现都有自己的依赖关系,由Simple Injector连接。

我不确定SendEmailCommand将如何确定要使用的实施(IEmailService)或如何使用运行时数据连接这些服务。

一些代码:

SendEmailCommand.cs 处理程序

public void Handle(SendEmail command) {
  IEmailService service;

  if(/* some condition */) {
    service = // What to do here?
  } else {
    service = // What to do here?
  }
}

服务可以用我猜测的策略代替。

1 个答案:

答案 0 :(得分:3)

最明显的解决方案(对我而言)是定义一个新的抽象,允许在向其提供运行时数据的同时调度到邮件服务实现。这可能如下所示:

interface IMailServiceDispatcher
{
    void Send([runtime data], [mail service arguments]);
}

这允许您的命令处理程序变为:

private readonly IEmailServiceDispatcher dispatcher;
public void Handle(SendEmail command) {
  var data = /* runtime data */
  this.dispatcher.Send(data, /* mail service parameters */);
}

现在,您可以为IMailServiceDispatcher创建一个可能如下所示的实现:

class EmailServiceDispatcher : IEmailServiceDispatcher
{
    IMailService man; IMailService smtp;
    public EmailServiceDispatcher(IMailService man, IMailService smtp) { ... }

    public void Send(runtimeData, parameters) =>
        GetService(runtimedData).Send(parameters);

    private IMailService GetService(runtimeData) =>
        /* some condition using runtime data */ ? man : smtp;
}

您可以按照以下方式连接所有内容:

container.Register<IMailServiceDispatcher, MailServiceDispatcher>();
container.RegisterConditional<IMailService, MandrillEmailService>(WithParamName("man"));
container.RegisterConditional<IMailService, SmtpEmailService>(WithParamName("smtp"));

// Helper method
private static Predicate<PredicateContext> WithParamName(string name) =>
    c => c.Consumer.Target.Name == name;