在Controller之前我们先回顾一下HttpMessageHandler通道。

在这个图中我留一个HttpContollerDispatcher没有说明。因为这个类也是继承自HttpMessageHandler,但由于HttpRoutingDispatcher已经不继承DelegatingHandler,所以就没有把HttpControllerDisPatcher作为HttpMessageHandler通道的最后一个节点。

HttpControllerDispather的主要功能就是根据请求路径筛选出Contoller,并激活运行。

HttpController

在程序中默认创建的Controller都继承自抽象类ApiController,但在WebAPI只要Controller实现接口IHttpController即可。

public interface IHttpController

{ 

Task<System.Net.Http.HttpResponseMessage> ExecuteAsync(HttpControllerContext controllerContext, CancellationToken cancellationToken); 

}

IHttpController只有一个ExecuteAsync方法,从方法的返回值。

HttpControllerDispatcher的SendAsync方法可以分成4个步骤:

  1. 获取HttpControllerDescriptor
  2. 创建HttpController
  3. 获取Controller上下文(HttpControllerContext)
  4. 执行HttpController的ExecuteAsync方法

获取HttpControllerDescriptor

HttpControllerDescriptor封装了某个HttpController的元数据(ControllerName,ControllerType)。ControllerType是HttpController对应的类型,因为有了这些信息,所以HttpControllerDescriptor就有了创建HttpController的能力

public class HttpControllerDescriptor

 { 

public HttpControllerDescriptor(); 

public HttpControllerDescriptor(HttpConfiguration configuration, string controllerName, Type controllerType); 

public HttpConfiguration Configuration { get; set; } 

public string ControllerName { get; set; } 

public Type ControllerType { get; set; } 

 }

整个获取HttpControllerDescriptor可以看成是根据路由变量controller的值在程序集中查找HttpControllerDescriptor。

WebAPI提供了查找的"标准化组件":HttpControllerSelector<IHttpControllerSelector,DefalutHttpControllerSelector>。

public interface IHttpControllerSelector

 { 

IDictionary<string, HttpControllerDescriptor> GetControllerMapping(); 

HttpControllerDescriptor SelectController(HttpRequestMessage request); 

 }

GetControllerMapping返回所有的HttpControllerDescriptor, SelectController按ControllerName返回ControllerType。

当然出于效率的考虑,WebAPI不会每次都去程序集中去查找。也是会做相应的缓存。对于加载缓存由两个"标准化组件"完成:IAssembliesResolver,IHttpControllerTypeResolver

IAssembliesResolver用于加载程序集。ASP.NET WebAPI注册是WebHostAssembliesResolver ,它定义在System.Web.Http.WebHost.dll中。它几乎返回运行过程中所有的程序集。

IHttpControllerTypeResolver用于从IAssembliesResolver获取到的程序集中筛选出所有的HttpController。WebAPI对它的默认实现是DefaultHttpControllerTypeResolver。筛选HttpController要满足如下条件:

  1. 直接或间接实现IHttpController的类
  2. 该类是公开的,非抽象的
  3. 该类的类名应该是以Controller结束

创建Controller

在得到HttpControllerDescriptor之后,HttpControllerDescriptor自身就有CreateController方法去创建Controller,但具体的工作HttpControllerDescriptor交给了HttpControllerActivator完成。

public interface IHttpControllerActivator

 { 

IHttpController Create(HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor, Type controllerType); 

 }

HttpControllerActivator是作为"标准化组件"注册在ServicesContainer中的.WebAPI的默认实现是DefaultHttpControllerActivator。

在DefaultHttpControllerActivator中最优先使用的并不是反射,而是采用的DependencyResolver方式进行创建。只有当DependencyResolver返回空是才采用反射方式创建HttpController,上一篇我也提到过HttpConfiguration的DependencyResolver默认是EmptyResolver,所以实际上WebAPI默认采用反射的方式创建HttpController。

当然我们可以自己去实现DependencyResolver。在Demo中我用Unity自大做了一个简单的UnityDependencyResolver。

Controller上下文

Controller上下文对应的类的是HttpControllerContext,ApiController的ControllerContext就是HttpControllerContext对象。 ControllerContext包含了请求中的各种信息。激活Controller最后一步就是根据请求信息生成ControllerContext并赋值给ApiController.ControllerContext。

public class HttpControllerContext

 { 

public HttpControllerContext(); 

public HttpControllerContext(HttpConfiguration configuration, IHttpRouteData routeData, HttpRequestMessage request); 

public HttpControllerContext(HttpRequestContext requestContext, HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor, IHttpController controller); 

public HttpConfiguration Configuration { get; set; } 

public IHttpController Controller { get; set; } 

public HttpControllerDescriptor ControllerDescriptor { get; set; } 

public HttpRequestMessage Request { get; set; } 

public HttpRequestContext RequestContext { get; set; } 

public IHttpRouteData RouteData { get; set; } 

 }

在ApiController中有一Initialize方法,该方法就是用于设置ControllerContext并标识ApiControlle已完成初始化(当ApiController未标识完成时会抛出InvalidOperationException异常)。所以我们在重写该方法时一定不要漏掉base.Initialize。

protected virtual void Initialize(HttpControllerContext controllerContext)

整个Contoller的激活大概如下流程:

源码

Github: https://github.com/BarlowDu/WebAPI (API_9)

ASP.NET WebAPI 09 Controller的激活的更多相关文章

  1. ASP.NET MVC Controller的激活

    最近抽空看了一下ASP.NET MVC的部分源码,顺带写篇文章做个笔记以便日后查看. 在UrlRoutingModule模块中,将请求处理程序映射到了MvcHandler中,因此,说起Controll ...

  2. Asp.net mvc 中的 Controller 的激活

    Controller 激活是指根据路由系统解析出来的 Controller 的名称创建 控制器(Controller)的过程,这里的控制器泛指实现了 IController 接口的类型 激活过程中的核 ...

  3. ASP.NET MVC——Controller的激活

    Controller的激活是根据在路由过程得到的Controller名称来创建对应的Controller对象.相关类如图: Controller激活的过程可通过如下序列图表示: 代码示例如下: str ...

  4. 重温ASP.NET WebAPI(二)进阶

    重温ASP.NET WebAPI(二)进阶   介绍 本文为个人对WebApi的回顾无参考价值. 本文内容: Rest和UnitOfWork 创建WebAPi的流程 IOC-Unity的使用 MEF ...

  5. Asp.Net WebApi核心对象解析(上篇)

    生活需要自己慢慢去体验和思考,对于知识也是如此.匆匆忙忙的生活,让人不知道自己一天到晚都在干些什么,似乎每天都在忙,但又好似不知道自己到底在忙些什么.不过也无所谓,只要我们知道最后想要什么就行.不管怎 ...

  6. ASP.NET WebApi 文档Swagger深度优化

    本文版权归博客园和作者吴双本人共同所有,转载和爬虫请注明博客园蜗牛原文地址,cnblogs.com/tdws   写在前面 请原谅我这个标题党,写到了第100篇随笔,说是深度优化,其实也并没有什么深度 ...

  7. ASP.NET WebApi 文档Swagger中度优化

    本文版权归博客园和作者吴双本人共同所有,转载和爬虫请注明原文地址:www.cnblogs.com/tdws   写在前面 在后台接口开发中,接口文档是必不可少的.在复杂的业务当中和多人对接的情况下,简 ...

  8. ASP.NET WEBAPI 的身份验证和授权

    定义 身份验证(Authentication):确定用户是谁. 授权(Authorization):确定用户能做什么,不能做什么. 身份验证 WebApi 假定身份验证发生在宿主程序称中.对于 web ...

  9. Enable Cross-Origin Requests in Asp.Net WebApi 2[Reprint]

    Browser security prevents a web page from making AJAX requests to another domain. This restriction i ...

随机推荐

  1. Android中GridView使用总结

    1.http://blog.csdn.net/hellogv/article/details/4567095  基础篇,GridView最基本的用法 2.http://my.eoe.cn/cainia ...

  2. 关于 Visual Studio 调试 Global 的一点总结

    在开发 MVC 的项目中遇到了些问题,想通过调戏查看找问题的原因,发现无法调试 Global 中的 Application_Start 方法,在网上找遍了也没有相应的解决办法,在经过了很多次尝试之后仍 ...

  3. ivqBlog 开源博客 (angularjs + express + mongodb)

    转向做全职前端差不多一年的时间了,其中学习了构建工具grunt,gulp,angularjs,coffeescript,less,sass,自己想要做全栈开发,所以自学了mongodb,nodejs, ...

  4. CORS(跨来源资源共享协议) 与 http 302状态

    昨天遇到的问题 使用ajax请求一个支持CORS的跨域页面(A),此页面返回302状态并且重新定向到页面(B).此时ajax停止不前,并且触发 ajax onerror 事件. 正确的相应应该是:aj ...

  5. studio 快捷键

    一,基础快捷键 1.1 Ctrl+N,Navigate | Class,快速打开某个类 1.2 Ctrl+Shift+N,  Navigate | File,  快速打开某个文件 1.3 Ctrl+S ...

  6. Ubuntu 中搭建 LAMP 及 php 开发工具

    所谓 LAMP,指的是:Linux+Apache+Mysql+Php 仅以此文做一个备忘录 Step1. 安装 Apache 1. 在 terminal 中输入一下命令并执行: sudo apt-ge ...

  7. HDU 3487 Play with Chain(Splay)

    题目大意 给一个数列,初始时为 1, 2, 3, ..., n,现在有两种共 m 个操作 操作1. CUT a b c 表示把数列中第 a 个到第 b 个从原数列中删除得到一个新数列,并将它添加到新数 ...

  8. 【Android学习】数据传递三种方式

    1.Application 注意在清单文件中的Application节点下注册android:name属性, 继承Application类,重写onCreate方法, 使用数据时,实例化自定义类时需要 ...

  9. Android样式的开发:Style篇

    前面铺垫了那么多,终于要讲到本系列的终篇,整合所有资源,定义成统一的样式.哪些该定义成统一的样式呢?举几个例子吧: 每个页面标题栏的标题基本会有一样的字体大小.颜色.对齐方式.内间距.外间距等,这就可 ...

  10. c/c++:重载 覆盖 隐藏 overload override overwrite

    http://www.cnblogs.com/qlee/archive/2011/07/04/2097055.html 成员函数的重载.覆盖与隐藏成员函数的重载.覆盖(override)与隐藏很容易混 ...