5

我正在使用 Simple Injector 作为我的 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>();
}
4

1 回答 1

3

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

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

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

Simple Injector 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);
于 2013-02-04T17:59:52.760 回答