使用Ninject的.NET MVC3服务定位器/依赖性解析器问题

Khe*_*pri 5 c# dependency-injection ninject service-locator asp.net-mvc-3

我有我认为的标准.NET MVC3存储库模式项目,我一直在玩/学习.这是非常标准的结构.

  • 存储库项目(使用下面提到的缓存基础结构)
  • 领域模型项目
  • 服务层项目
  • MVC演示项目

我遇到了一个场景,我需要注入一个类的私有成员,该类只有一个静态构造函数,这使得构造函数注入失败.

有问题的类是使用我刚刚完成的AppFabric缓存实现的包装器.(对于那些如此倾向的人,我的实施基于 http://cgeers.wordpress.com/2010/07/04/windows-server-appfabric-caching/)

基本上我有:

  • 接口ICacheProvider
  • class DefaultCacheProvider:ICacheProvider
  • static class Cache(利用我注入的任何实现)

静态类缓存是我想要注入一个解析为DefaultCacheProvider的ICacheProvider的地方.

    private static readonly ICacheProvider CacheProvider;

    static Cache()
    {
        //DependencyResolver.Current.GetService<ICacheProvider>();

        //CacheProvider =
        //    (ICacheProvider)ServiceLocator.Current
        //                            .GetInstance(typeof(ICacheProvider));
    }

    public static void Add(string key, object value)
    {
        CacheProvider.Add(key, value);
    }

    public static void Add(string key, object value, TimeSpan timeout)
    {
        CacheProvider.Add(key, value, timeout);
    }

    public static object Get(string key)
    {
        return CacheProvider[key];
    }

    public static bool Remove(string key)
    {
        return CacheProvider.Remove(key);
    }
Run Code Online (Sandbox Code Playgroud)

根据我所读到的内容,这似乎是ServiceLocator的一个场景,但我已经看到了一些非常强烈的意见(反模式等等),而且我对它的熟悉程度很低,所以我不确定可行的实施.

我已经看到StackOverflow上的建议将Cache类设计为标准类并在SingletonScope中注入ICacheProvider

kernel.Bind<ICacheProvider>().To<DefaultCacheProvider>().InSingletonScope();
Run Code Online (Sandbox Code Playgroud)

但我个人更喜欢静态包装,方便使用.

ServiceLocator是设置到这里的方式还是还有一些我不知道的明显的东西?如果要使用ServiceLocator,是否可以使用与Ninject的搭配?我知道Ninject现在有服务定位器功能,但不确定如何实现.

谢谢你的任何信息.

Ed *_*pel 5

我认为你的方法缺少控制反转容器的本质来提供依赖注入.

根据我的阅读,这似乎是ServiceLocator的一个场景,但我已经看到了一些非常强烈的意见(反模式等)

非常强烈的观点通常包括对Singleton模式的厌恶,换句话说,使用静态类来提供服务.这里的问题是Cache你编写的类是与你引用的反模式相同的Singleton模式.

消耗Cache单例的代码是什么样的?让我提出一个假设.

public class SomeClass
{
    public string GetSomeMetaData()
    {
        return Cache.Get("magicKey");
    }
}
Run Code Online (Sandbox Code Playgroud)

在这种情况下,您通过使用Singleton抽象了IoC并避免了DI.我会建议

public class SomeClass
{
    private readonly ICacheProvider _cacheProvider;

    public SomeClass(ICacheProvider cacheProvider)
    {
        _cacheProvider = cacheProvider;
    }

    public string GetSomeMetaData()
    {
        return _cacheProvider.Get("magicKey");
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,消费ICacheProvider直接发生在需要它的类中,并且可以更容易地适应ICacheProvider实现的变化.它具有简化测试的额外好处.Singleton模式几乎不可能进行测试.