简单注射器混合生活方式,适用于每个网络请求和生命周期范围

本文关键字:请求 网络 求和 生命 范围 周期 混合 注射器 生活方式 适用于 简单 | 更新日期: 2023-09-27 17:57:21

我正在使用简单注入器作为我的 IoC 容器,并采用以下技术为某些对象注册为每个 Web 请求或每个线程的"混合"生活方式。

interface IUnitOfWork { }
interface IWebUnitOfWork : IUnitOfWork { }
interface IThreadUnitOfWork : IUnitOfWork { }
class UnitOfWork : IWebUnitOfWork, IThreadUnitOfWork { }
container.RegisterPerWebRequest<IWebUnitOfWork, UnitOfWork>();
container.RegisterLifetimeScope<IThreadUnitOfWork, UnitOfWork>();
container.Register<IUnitOfWork>(() => container.GetInstance<UnitOfWork>());
// Register as hybrid PerWebRequest / PerLifetimeScope.
container.Register<UnitOfWork>(() =>
{
    if (HttpContext.Current != null)
        return container.GetInstance<IWebUnitOfWork>() as UnitOfWork;
    else
        return container.GetInstance<IThreadUnitOfWork>() as UnitOfWork;
});

我对这个解决方案并不完全满意,因为对于每个要求,我必须定义额外的空接口才能使其工作并确保它们被我的具体类引用。

有什么理由我不应该使用以下扩展方法而不是定义额外的接口吗?如果这些方法存在一些问题,是否有其他方法可以完全自信地确定我的容器的当前实例正在 IIS 中运行?

public static void RegisterHybridLifestyle<TService, TImplementation>(
    this Container container)
    where TService : class
    where TImplementation : class, TService
{
    if (System.Web.Hosting.HostingEnvironment.ApplicationHost != null)
        container.RegisterPerWebRequest<TService, TImplementation>();
    else
        container.RegisterLifetimeScope<TService, TImplementation>();
}
public static void RegisterForLifestyle<TConcrete>(
    this Container container)
    where TConcrete : class
{
    if (HostingEnvironment.ApplicationHost != null)
        container.RegisterPerWebRequest<TConcrete>();
    else
        container.RegisterLifetimeScope<TConcrete>();
}

更新

上述问题和后续问题是基于对SimpleInjector和混合注册的误解。上面和 SO 上其他地方描述的技术适用于容器可以为 Web 请求和未在 Web 请求上下文中运行的后台进程的请求提供服务的情况。我一直试图实现的是变量注册,以满足适用于 Web 请求线程请求的容器配置。即我需要将我的容器配置为在 IIS 中工作并在 Windows 服务中工作。我不需要可以同时满足两者的动态注册。

这样做的结果是以下扩展方法,我已经从我的解决方案中删除了"额外"接口:-)

public static void RegisterForScope<TService, TImplementation>(this Container container)
    where TService : class
    where TImplementation : class, TService
{
    if (System.Web.Hosting.HostingEnvironment.ApplicationHost != null)
        container.RegisterPerWebRequest<TService, TImplementation>();
    else
        container.RegisterLifetimeScope<TService, TImplementation>();
}
public static void RegisterForScope<TConcrete>(this Container container)
    where TConcrete : class
{
    if (System.Web.Hosting.HostingEnvironment.ApplicationHost != null)
        container.RegisterPerWebRequest<TConcrete>();
    else
        container.RegisterLifetimeScope<TConcrete>();
}

简单注射器混合生活方式,适用于每个网络请求和生命周期范围

我对这个解决方案并不完全满意

是的,我同意这一点。老实说,不得不做这样的事情实际上很糟糕 IMO。这就是为什么在Simple Injector 2.0中修复了这个问题。它包含生活方式的明确概念,并且将包含一种Lifestyle.CreateHybrid方法,这使得注册混合生活方式变得更加容易。

但是,您似乎不需要混合生活方式。混合生活方式是一种可以动态切换的生活方式(每次调用GetInstance<T>和每次注射),而您似乎只需要在启动期间切换。我认为使用您定义的RegisterHybridLifestyle扩展方法没有害处,但请记住,这不是真正的混合生活方式(因此名称有点误导),而只是一个配置/部署开关。

简单的注射器 2 及更高版本使这变得更加容易,它将允许您执行以下操作:

// Define a lifestyle once based on the deployment.
Container.Options.DefaultScopedLifestyle =
    Lifestyle.CreateHybrid(
        lifestyleSelector: HostingEnvironment.ApplicationHost != null,
        trueLifestyle: new WebRequestLifestyle(),
        falseLifestyle: new LifetimeScopeLifestyle());
// And use it for registering the unit of work
// (no extra interfaces needed anymore).
container.Register<IUnitOfWork, UnitOfWork>(Lifestyle.Scoped);
// After setting DefaultScopedLifestyle, you can usr Lifestyle.Scoped
container.RegisterCollection(
    typeof(ISubscriber<>),
    new [] { typeof(ISubscriber<>).Assembly },
    Lifestyle.Scoped);