c# - 如何使用 WebAPI、WCF、SignalR 和后台任务在 MVC Web 应用程序中配置简单的注入(inject)器容器和生活方式

标签 c# dependency-injection simple-injector

简单的注入(inject)器文档提供了关于如何为 WebRequest、Web API、WCF 等设置容器的很好的示例,但这些示例一次只针对一种技术/生活方式。我们的 Web 应用程序同时使用了其中的大部分! 我不清楚如何配置容器以适应多种生活方式。


假设我有一个带有 Web API 的 MVC 项目。我有以下对象:

  • MyDbContext :我的实体代码优先数据库上下文
  • 由 MyDataProvider 实现的 IMyDataProvider : 包含查询逻辑并使用 MyDbContext
  • MyController:使用 IMyDataProvider 的 MVC Controller
  • MyApiController : 使用 IMyDataProvider 的 WebApi Controller

我应该为每种生活方式创建和配置一个容器吗?

当我用 RegisterPerWebRequest<T> 注册所有内容时适用于两种类型的 Controller 。这样安全吗?还是在 Web API Controller 中使用 async/await 时会遇到麻烦?

当我同时拥有注入(inject)相同实例的 MVC 和 Web API Controller 时,最佳配置是什么?

我应该采用混合生活方式吗?


现在让事情复杂化......我们的应用程序还使用后台任务和 SignalR。

这两者有时会发生在 WebRequest 之外,并且需要访问与上述相同的对象。

最好的解决方案是使用生命周期范围?

我需要为这种生活方式创建一个新容器吗?或者我可以重用/重新配置我的 MVC/Web API 容器吗?

是否存在三重生活方式?

最佳答案

我不得不说,前段时间我偶然发现了一个类似的场景,我最终通过我的网络 API 和 signalR 共享了我的配置,但是你需要为 signalR 实现一个自定义的生活方式,因为它不是基于网络请求的。

特别是在 signalR 中,您会发现在处理 Hub 中的每个 Web 请求依赖项时存在一些问题,其中一些将变为 null,例如 httpContext.Current 等。

解决方案:

您需要 WebRequestLifeSTLye 和 Lifestyle.Transient、Lifestyle.Singleton 或 LifetimeScopeLifestyle 之间的混合生活方式。我结束了我使用装饰器模式,你可以阅读这个 post还有这个post .

我的装饰师

public class CommandLifetimeScopeDecorator<T> : ICommandHandler<T>
    {
        private readonly Func<ICommandHandler<T>> _handlerFactory;
        private readonly Container _container;

        public CommandLifetimeScopeDecorator(
        Func<ICommandHandler<T>> handlerFactory, Container container)
        {
            _handlerFactory = handlerFactory;
            _container = container;
        }

        public void Handle(T command)
        {
            using (_container.BeginLifetimeScope())
            {
                var handler = _handlerFactory(); // resolve scoped dependencies
                handler.Handle(command);
            }
        }

    }

    public interface ICommandHandler<in T>
    {
        void Handle(T command);
    }

我使用 signalR 的集线器激活器管理依赖关系

public class MyHubActivator : IHubActivator
    {
        private readonly Container _container;

        public MyHubActivator(Container container)
        {
            _container = container;
        }

        public IHub Create(HubDescriptor descriptor)
        {
            return _container.GetInstance(descriptor.HubType) as IHub;
        }
    }

一个复合根文件,您将在其中处理您的依赖项

public CompositRoot(Container container)
{
    _container = container;
}
public container Configure()
{
   // _container.Registerall container dependencies
   return _container;
}

然后在引导您的应用程序时共享您的复合根配置

var compositRoot = new CompositRoot(simpleInjector.Container); //simple injector instance
compositRoot.Configure();

对于信号R

GlobalHost.DependencyResolver.Register(typeof(IHubActivator), () => new MyHubActivator(compositRoot));

并且您可以在其他项目中重用您的配置!

我的两分钱 希望对您有所帮助!

关于c# - 如何使用 WebAPI、WCF、SignalR 和后台任务在 MVC Web 应用程序中配置简单的注入(inject)器容器和生活方式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26433012/

相关文章:

c# - 如果一个测试没有失败,则运行所有测试

Spring MVC 2.5 使用注释和 XML 配置的混合,但 xml 被忽略

c# - 从支付网关重定向后丢失用户 cookie 和 session 数据

dependency-injection - 我的 Unity DI 容器中所有对象的列表

php - Symfony 4 Custom Bundle 使用 Logger Interface

mapping - 简单Injector注册IMappingEngine(AutoMapper)

c# - 重构 C# 保存命令处理程序

c# - 在简单注入(inject)器中调用拦截的方法调用上的函数

c# - 使用 linq 更新对象集合中的属性

c# - 使用 PooledRedisClientManager() 连接到 Ubuntu 虚拟机上的远程 Redis 实例