一. 为什么要进行扩展

  在前面的分析中,我们知道默认的Controller激活系统只能实例化无参构造函数的Controller类型,但在某些情况一下,我们希望某些服务的实例能够自动注入到Controller实例中,从而达到服务接口和实现的隔离,减小重复的代码,提高系统的可维护性和灵活性。也就是说我们希望在Controller激活中引入依赖注入。关于依赖注入的概念这里就不解释了,请自行查询相关的资料。基于.net依赖注入框架也有很多,下面的例子主要使用微软企业库的Unity。

  在上一节的分析中,我们知道Controller的激活实际是包括获取IControllerFacotry和IController实例, 在这两个级别都可以引入扩展实现在我们的目的,现在具体来看一下。

一.实列化IControllerFactory  Level

  在这个Level我们知道了路由系统提供的路由信息,当然甚至可以从头到尾自定义实现一个ControllerFactory,包括Controller类型的确定和实例化,但通常没什么必要。我们目标是在Controller实例化阶段注入服务实例,主要还是在Controller实例化阶段。

二. 实列化Controller Level

  在这一阶段,我们已经确定Controller的类型,通过上一节的分析,有三个点我们可以引入依赖注入。我们假设有一个要显示所有Customer信息的页面,在其中应用了仓储模式,在CustomerController实例化进自动注入仓储的实例。

1. 继承DefaultControllerFactory,重写GetControllerInstance方法。大概的代码如下:

  

  1. public class UnityControllerFactory : DefaultControllerFactory
  2.  
  3.   {
  4.     public IUnityContainer Container
  5.     {
  6.       get;
  7.       private set;
  8.     }
  9.  
  10.     public UnityControllerFactory(IUnityContainer container)
  11.     {
  12.       this.Container = container;
  13.     }
  14.  
  15.     protected override IController GetControllerInstance(RequestContext requestContext, Type controllerType)
  16.     {
  17.       return (IController)this.Container.Resolve(controllerType);
  18.     }
  19.   }

  然后在Global.asax的Application_Start中注册

  1. private void RegisterUnityControllerFactory()
  2.  
  3.   {
  4.     IUnityContainer container = new UnityContainer();
  5.     container.RegisterType<ICustomerRepository, CustomerRepository>();
  6.     ControllerBuilder.Current.SetControllerFactory(new UnityControllerFactory(container));
  7.   }

  2. 自定义实现IControllerActivator。大概的代码如下:

  

  1. public class UnityControllerActivator : IControllerActivator
  2.  
  3.   {
  4.     public IUnityContainer Container
  5.     {
  6.       get;
  7.       private set;
  8.     }
  9.  
  10.     public UnityControllerActivator(IUnityContainer c)
  11.     {
  12.       this.Container = c;
  13.     }
  14.  
  15.     public IController Create(RequestContext requestContext, Type controllerType)
  16.     {
  17.       return (IController)this.Container.Resolve(controllerType);
  18.     }
  19.   }

  注册代码:

  

  1.  private void RegisterUnityControllerActivator()
  2.  
  3.    {
  4.     IUnityContainer container = new UnityContainer();
  5.     container.RegisterType<ICustomerRepository, CustomerRepository>();
  6.     ControllerBuilder.Current.SetControllerFactory(new DefaultControllerFactory(new UnityControllerActivator(container)));
  7.   }

  3.自定义实现全局的IDependencyResolver

  1. public class UnityDependencyResolver : IDependencyResolver
  2. {
  3. private IUnityContainer Container
  4. {
  5. get;
  6. set;
  7. }
  8.  
  9. public UnityDependencyResolver()
  10. {
  11. this.Container = new UnityContainer();
  12. }
  13.  
  14. public UnityDependencyResolver RegisterType<TFrom, TTo>() where TTo : TFrom
  15. {
  16. this.Container.RegisterType<TFrom, TTo>();
  17. return this;
  18. }
  19.  
  20. public object GetService(Type serviceType)
  21. {
  22. try
  23. {
  24. return this.Container.Resolve(serviceType);
  25. }
  26. catch(ResolutionFailedException)
  27. {
  28. return null;
  29. }
  30. }
  31.  
  32. public IEnumerable<object> GetServices(Type serviceType)
  33. {
  34. try
  35. {
  36. return this.Container.ResolveAll(serviceType);
  37. }
  38. catch (ResolutionFailedException)
  39. {
  40. return null;
  41. }
  42. }

    注册代码:

  1. private void SetDependencyResolver()
  2. {
  3. UnityDependencyResolver resolver = new UnityDependencyResolver();
  4. resolver.RegisterType<ICustomerRepository, CustomerRepository>();
  5. DependencyResolver.SetResolver(resolver);
  6. }

三.真的要扩展吗?

  天下没有完美的东西,引入一样东西我们必须要充分认识它的优势,也要认清它的副作用。在实际项目中,个人觉得大部分情况是没必要在Controller这个Level引入依赖注入,在Controller的Action中,通常我们要调用业务逻辑服务,业务逻辑的服务接口通常是没必要再引入一层另外的抽象。

  另外,在这里大致总结一下,在Action中大概有两种方式来调用业务逻辑,一种利用Command模式,即把每个Action 对一个Command,在Command中再调用业务逻辑服务.一种是每个Controller包含一个或多个粗粒度的业务包装服务类,每个Action中直接调用一个或多个服务类处理。具体使用那一种模式,应根据你的项目情况来决定,这里不展开细说了。

最后所有的测试代码可以在这里下载http://files.cnblogs.com/jjyjjyjjy/TestAsp_Net_Mvc_ControllerActivator_DI.rar

ASP.NET MVC4学习笔记之Controller激活的扩展的更多相关文章

  1. ASP.NET MVC4学习笔记之Controller的激活

    一. 高层相关类说明 当路由系统根据请求Url收集路由信息后,下一步就要将路由信息传给Controller激活系统,Controller激活系统负责实现了IController接口的Controlle ...

  2. ASP.NET MVC 学习笔记-7.自定义配置信息 ASP.NET MVC 学习笔记-6.异步控制器 ASP.NET MVC 学习笔记-5.Controller与View的数据传递 ASP.NET MVC 学习笔记-4.ASP.NET MVC中Ajax的应用 ASP.NET MVC 学习笔记-3.面向对象设计原则

    ASP.NET MVC 学习笔记-7.自定义配置信息   ASP.NET程序中的web.config文件中,在appSettings这个配置节中能够保存一些配置,比如, 1 <appSettin ...

  3. asp.net mvc4 学习笔记一(基本原理)

    做了8年的asp.net webform,用过MVVM但还没用过MVC , 虽然项目不用MVC,但是还是想了解一下,今天第二天学习,以下是学习心得. VS2012默认带有asp.net mvc3和as ...

  4. ASP.NET MVC4学习笔记路由系统概念与应用篇

    一.概念 1.路由是计算机网络中的一个技术概念,表示把数据包从一个网段转发至另一网段.ASP.NET中的路由系统作用类似,其作用是把请求Url映射到相应的"资源"上,资源可以是一段 ...

  5. ASP.NET MVC4学习笔记之总体概述

    断断续续使用ASP.NET MVC框架也有一年多了,也算积累了一些经验,唉,一直想写一些笔记好好总结一下,人太懒不想动笔,今天终于决定开始.希望自己能坚持下去. 这篇文章大体介绍ASP.NET MVC ...

  6. ASP.NET MVC5学习笔记之Controller同步执行架构分析

    在开始之前,声明一下,由于ASP.NET MVC5正式发布了,后面的分析将基于ASP.NET MVC5最新的源代码.在前面的内容我们分析了怎样根据路由信息来确定Controller的类型,并最终生成C ...

  7. ASP.NET MVC4学习笔记路由系统实现

    一.路由实现 路由系统实际是一个实现了ASP.NET IHttpModule接口的模块,通过注册HttpApplication的PostResolveRequestCache 事件对Url路由处理.总 ...

  8. ASP.NET MVC4学习笔记

    一.MVC简介

  9. ASP.NET MVC5学习笔记之Controller执行ControllerDescriptor和ActionDescriptor

    一. ControllerDescriptor说明 ControllerDescriptor是一个抽象类,它定义的接口代码如下: public abstract class ControllerDes ...

随机推荐

  1. Unity小厨房之-----背后视角摄像机

    相信每一个接触过Unity的人,应该都认识Unity引擎自带的那个水管工,也一定知道那个小人模型上挂着3个脚本控制着小人的一切,今天我就来说一下关于Unity背后视角摄像机的实现,当然方法并不唯一,这 ...

  2. Linux平台块设备到字符设备(裸设备)的三种映射方式(转载)

    在Linux平台oracle rac的组建过程中,如果使用ASM+RAW的存储方式的话,由于asm不支持块设备,支持持字符访问设备,所以需要配置将Block Device Drive转变成Charac ...

  3. Hibernate三种状态的区分,以及save,update,saveOrUpdate,merge等的使用 引自http://www.blogjava.net/TiGERTiAN/archive/2008/10/25/236519.html

    Hibernate的对象有3种状态,分别为:瞬时态(Transient). 持久态(Persistent).脱管态(Detached).处于持久态的对象也称为PO(Persistence Object ...

  4. OC基础(19)

    类扩展(Class Extension) Block基本概念 typedef和Block Block注意事项 *:first-child { margin-top: 0 !important; } b ...

  5. 翻译:Knockout 轻松上手 - 1 Knockout 是什么?

    原文名称:KnockoutJS Starter Knockout 是一个非常棒的脚本库,可是我发现许多人并不了解它,所以,思胜翻译了这本著作. 这一节,我们将会讨论 Knockout 可以做什么,又如 ...

  6. CODESOFT 2015中的二维码该怎样生成

    由于二维条码具有储存量大.保密性高.追踪性高.抗损性强.备援性大.成本便宜等特性,其应用 渐趋广泛,因此二维码的制作对于CODESOFT条码设计软件的用户来讲可谓司空见惯.我们最常见的二维码要数QR码 ...

  7. python实现批量ping IP,并将结果写入

    最近工作需要,写了一个Python小脚本,分享给大家,因为公司的IP用的差不多了,然后离职人员的IP有没有及时删除,导致没多少IP用了,所以做了一个python脚本跑了跑,清出来一堆ping不通的IP ...

  8. 【LeetCode】18. 4Sum

    题目: 思路:这题和15题很像,外层再加一个循环稍作修改即可 public class Solution { public List<List<Integer>> fourSu ...

  9. Windows7:Visual Studio 2008试用版的评估期已经结束解决方法

    Windows7:Visual Studio 2008试用版的评估期已经结束解决方法       以前在Windows2003碰到这个问题时,都是到"控制面板→添加或删除程序"选择 ...

  10. 【Qt 】QSettings写ini配置文件

    QSettings写ini配置文件(Qt版本5.2): #include "inidemo.h" #include <QSettings> #include <Q ...