MVC3中的Ninject依赖注入 - 控制器之外

时间:2011-11-11 16:57:08

标签: c# .net asp.net-mvc dependency-injection ninject

我们在MVC3项目中使用Ninject来进行依赖注入。 我使用NuGet为Ninject和Ninject.MVC3包添加包引用。当我这样做时,它在我的App_Start文件夹中创建了一个NinjectMVC3类:

public static class NinjectMVC3
{
    private static readonly Bootstrapper bootstrapper = new Bootstrapper();

    public static void Start()
    {
        DynamicModuleUtility.RegisterModule(typeof(OnePerRequestModule));
        DynamicModuleUtility.RegisterModule(typeof(HttpApplicationInitializationModule));
        bootstrapper.Initialize(CreateKernel);
    }

    public static void Stop()
    {
        bootstrapper.ShutDown();
    }

    private static IKernel CreateKernel()
    {
        var kernel = new StandardKernel();
        RegisterServices(kernel);
        return kernel;
    }

    private static void RegisterServices(IKernel kernel)
    {           
        kernel.Bind<IPrincipal>().ToMethod(c => HttpContext.Current.User);
    }
}

到目前为止,这对于解决我的控制器中的依赖关系非常有用:

public class HomeController : Controller {
    protected IPrincipal principal { get; set; }

    public HomeController(IPrincipal principal) {
        this.principal = principal;
    }
}

这个控制器依赖于IPrincipal,我已经在我的bootstrapper类中设置了解析为HttpContext.Current.User。我有另一个类依赖于IPrincipal而不是控制器:

public class NonControllerClass
{
    protected IPrincipal Principal { get; set; }

    public NonControllerClass(IPrincipal principal) {
    }
}

我将如何解决此依赖关系?如果它不是构造函数的参数,我该怎么做?     }

1 个答案:

答案 0 :(得分:3)

嗯,理想情况下,这永远不应成为问题。所有依赖项都应该被引入控制器,并且这些依赖项依赖的依赖项也应该自动注入。

在MVC中,(几乎)一切都从控制器开始。所以你可能有:

public class HomeController : Controller { 
    protected IMyService myService { get; set; } 

    public HomeController(IMyService myService) { 
        this.myService = myService; 
    } 
} 

public class MyService {
    protected IPrincipal principal;

    public MyService(IPrincipal principal) { this.principal = principal)
}

请注意您不必做任何事情,您的服务会自动注入正确的依赖项,因为您的服务已注入您的控制器。

但是,有时您可能需要动态创建对象。在这种情况下,您可以使用MVC DependencyResolver。

var principal = DependencyResolver.Current.GetService<IPrincipal>();

除非绝对必要,否则您应该避免这样做,因为这被视为反模式(称为服务位置)。有时候你没有太多选择。

如果您不想使用构造函数注入,则可以使用属性注入。

public class MyService {
    [Inject]
    public IPrincipal principal {get; set;}
}