asp.net-mvc – 如何在Controller中使用依赖注入
我有下面的代码,没有任何问题
MAUserController.cs public class MAUserController : ApiController { ILogService loggerService; IMAUserService _service; public MAUserController(ILogService loggerService,IMAUserService Service) { this.loggerService = loggerService; this._service = Service; } } DependencyInstaller.cs public class DependencyInstaller : IWindsorInstaller { public void Install(IWindsorContainer container,IConfigurationStore store) { container.Register( Component.For<ILogService>().ImplementedBy<LogService>().LifeStyle.PerWebRequest,Component.For<IDatabaseFactory>().ImplementedBy<DatabaseFactory>().LifeStyle.PerWebRequest,Component.For<IUnitOfWork>().ImplementedBy<UnitOfWork>().LifeStyle.PerWebRequest,AllTypes.FromThisAssembly().BasedOn<IHttpController>().LifestyleTransient(),AllTypes.FromAssemblyNamed("ISOS.Health.Service").Where(type => type.Name.EndsWith("Service")).WithServiceAllInterfaces().LifestylePerWebRequest(),AllTypes.FromAssemblyNamed("ISOS.Health.Repository").Where(type => type.Name.EndsWith("Repository")).WithServiceAllInterfaces().LifestylePerWebRequest() ); } } 如果我使用普通的控制器而不是ApiController,那么它会给我一个错误 public class UserController : Controller { ILogService loggerService; IMAUserService _service; public UserController(ILogService loggerService,IMAUserService Service) { this.loggerService = loggerService; this._service = Service; } } 这会产生错误:
我使用CastleDI Windsor进行依赖注入. 我需要做任何事情或注册一些东西吗? 解决方法
第一种方法
建议:谨慎使用,因为它可能会导致Castle Windsor的内存泄漏. 您必须创建一个控制器激活器,它应该实现IControllerActivator接口,以便使用您的DI容器来创建控制器实例: public class MyWindsorControllerActivator : IControllerActivator { public MyWindsorControllerActivator(IWindsorContainer container) { _container = container; } private IWindsorContainer _container; public IController Create(RequestContext requestContext,Type controllerType) { return _container.Resolve(controllerType) as IController; } } 然后,将此类添加到DependencyInstaller: public class DependencyInstaller : IWindsorInstaller { public void Install(IWindsorContainer container,IConfigurationStore store) { container.Register( // Current code... Component.For<IControllerActivator>() .ImplementedBy<MyWindsorControllerActivator>() .DependsOn(Dependency.OnValue("container",container)) .LifestyleSingleton(); ); } } 另外,根据Windsor容器创建自己的依赖项解析器: public class MyWindsorDependencyResolver : IDependencyResolver { public MyWindsorDependencyResolver(IWindsorContainer container) { _container = container; } private IWindsorContainer _container; public object GetService(Type serviceType) { return _container.Resolve(serviceType); } public IEnumerable<object> GetServices(Type serviceType) { return _container.ResolveAll(serviceType).Cast<object>(); } } 然后,最后,在Global.asax.cs中的Application_Start方法中注册依赖项解析器: DependencyResolver.SetResolver(new MyWindsorDependencyResolver(windsorContainer)); 这样,当MVC需要控制器激活器通过它的依赖解析器时,它将获得我们的,它将使用我们的Windsor容器来创建具有所有依赖性的控制器. 为了避免使用IControllerActivator导致内存泄漏,最简单的解决方案是使用每个线程或每个Web请求的生活方式,而不是默认(单例),瞬态和池化,用于已注册的组件.有关如何使用Castle Windsor Container避免内存泄漏的详细信息,请查看this link. 第二种方法 然而,正如@PhilDegenhardt所指出的,更好更正确的方法是实现自定义控制器工厂,以便能够释放由Castle Windsor DI Container创建的控制器组件. Here you can find an example (see the section about Dependency Injection). 从该示例中可以看出,实现可能是: 的Global.asax.cs: public class MvcApplication : System.Web.HttpApplication { private WindsorContainer _windsorContainer; protected void Application_Start() { var _windsorContainer = new WindsorContainer(); _windsorContainer.Install( new DependencyInstaller(),// Other installers... ); ControllerBuilder.Current.SetControllerFactory(new WindsorControllerFactory(_windsorContainer.Kernel)); } protected void Application_End() { if (_windsorContainer != null) { _windsorContainer.Dispose(); } } } WindsorControllerFactory.cs: public class WindsorControllerFactory : DefaultControllerFactory { private readonly IKernel _kernel; public WindsorControllerFactory(IKernel kernel) { _kernel = kernel; } public override void ReleaseController(IController controller) { _kernel.ReleaseComponent(controller); // The important part: release the component } protected override IController GetControllerInstance(RequestContext requestContext,Type controllerType) { if (controllerType == null) { throw new HttpException(404,string.Format("The controller for path '{0}' could not be found.",requestContext.HttpContext.Request.Path)); } return (IController)_kernel.Resolve(controllerType); } } (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
- 我的ASP.NET Web应用程序无法“找到”App_Code文件夹中的任
- asp.net – 为什么这个JSON返回“无效的JSON原语”?
- asp.net – 如何更改AjaxControlToolkit HtmlEditorExtende
- asp.net-mvc – 如何使用razor语法转换为对象
- asp.net – 页面查看计数器像StackOverFlow
- asp.net-mvc – 模型绑定复选框问题列表
- asp.net-mvc – 在服务器上安装ASP.NET MVC 4
- asp.net-mvc-3 – 为什么在使用Azure缓存(.NET MVC3应用程序
- asp.net – 可以通过移动设备的Web浏览器上传图片吗?
- asp.net-mvc – 如何在主机上运行ASP.Net MVC应用程序?
- asp.net – 是否允许使用manifest.json的相对路径
- asp.net – BC30560:’ExtensionAttribute’在名
- asp.net – WebMethod未被Visual Studio 2013中的
- 通过ASP.NET菜单控件禁用javascript生成
- asp.net – 错误:数据绑定方法(如Eval(),XPath(
- asp.net – 隐藏表单值
- asp.net-mvc – 如何将Bootstrap下拉式样应用于A
- ASP.NET MVC Web应用程序中的控制器是否应该调用
- asp.net-mvc-3 – 创建MVC3剃刀助手,如Helper.Be
- asp.net-mvc – 发布ASP.NET MVC 3 Html.TextAre