我们在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) {
}
}
我将如何解决此依赖关系?如果它不是构造函数的参数,我该怎么做? }
答案 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;}
}