gpt4 book ai didi

c# - Simple Injector 注册和执行内部 CommandHandler

转载 作者:行者123 更新时间:2023-11-30 23:23:41 24 4
gpt4 key购买 nike

在我的开发团队中,我们正在使用 this article 中描述的 CQRS 模式并使用推荐的 DI 容器:Simple Injector (双关语)。

这是我们当前的项目结构:

  • 01 WebApp 引用 => [02] [03]
  • 02 契约(Contract)
  • 03 业务层

项目 01 是客户端(一个 ASP.NET MVC 4 应用程序),我们在其中使用 Simple Injector (Bootstrapper) 注册我们的应用程序服务。项目 03 是所有命令和查询处理程序定义的地方,如本文所述。项目 02 是定义命令和查询的地方。为了让 Simple Injector 注册处理程序,客户端可以直接引用业务层程序集。

容器像这样注册commandhandlers

container.Register(typeof(ICommandHandler<>), assemblies);

现在的问题是,我们的一位开发人员不小心忘记将他的处理程序类之一声明为 public。所以不是:

public class AddCustomerCommandHandler : ICommandHandler<AddCustomerCommand> { ... }

他写道:

class AddCustomerCommandHandler : ICommandHandler<AddCustomerCommand> { ... }

现在根据msdn ,如果省略访问修饰符,则使用内部:

Classes and structs that are declared directly within a namespace (in other words, that are not nested within other classes or structs) can be either public or internal. Internal is the default if no access modifier is specified.

现在内部定义为:

The type or member can be accessed by any code in the same assembly, but not from another assembly.

尝试执行此命令后,运行时不会抛出任何错误,命令会顺利执行。我对此感到惊讶,因为我预计 Simple Injector 会出现验证错误,但事实并非如此。此外,当尝试从客户端直接实例化处理程序对象时,编译器给我一个错误,提示 它无法访问此处的内部类,正如预期的那样!那么为什么 Simple Injector 能够在它的访问修饰符是 internal 的情况下注册这个 commandhandler?

最佳答案

在以前的版本中,Simple Injector 的批量注册默认跳过内部类型,并且有一个标志允许您也注册内部类型。

这种方法已被证明是有问题的,因为在某些应用程序中,缺少的类型允许应用程序继续运行,同时显示不正确的行为(因此静默失败)。

为防止这种情况发生,我们更改了此行为,现在您将始终看到这种类型也被注册。这个想法是,默默地跳过预期的类型比其他任何事情都糟糕得多。由于大多数应用程序以完全信任的方式运行,可以构造和解析内部类型,因此从 Simple Injector 的角度来看,类型是内部的就很好。对于其他应用程序类型,调用 Verify 将快速检测到不可构造的类型。

同样从应用程序的角度来看,内部类型应该没有问题,因为消费者与该类型实现的公共(public)接口(interface)对话。

在您的情况下失败的原因很可能是因为您分派(dispatch)了处理程序并在分派(dispatch)期间使用了 dynamic 类型。您所看到的是 IMO C# 动态基础结构中的一个怪癖。 C# 尝试使用反射找到您的 Handle 方法,但 Handle 方法是内部的,因为它的周围类型是内部的。它找不到此方法,即使该类型实现了包含该方法的公共(public)接口(interface)。这是一个怪癖,我相信 C# 仍然可以找到该方法;但事实并非如此。这就是您的代码失败的原因。

您可以做几件事来防止将来出现此类问题。例如,您可以定义一个单元测试来检查是否所有处理程序都是公共(public)的。或者您可以定义一个特殊的通用和公共(public)包装器类,将处理程序作为构造函数参数并解析该包装器而不是处理程序。然后您可以在该包装器上使用动态类型。或者您注册一个您确保公开的最外层装饰器。 C# 反射(reflect)了最外层的类型,因此这将起作用。

您还可以将检查集成到 Simple Injector 中,您可以在其中有条件地注册最外层的装饰器,并在处理程序是内部的情况下让谓词抛出异常。

关于c# - Simple Injector 注册和执行内部 CommandHandler,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38245451/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com