目录

  1. HttpController

  2. 创建HttpController

    • IAssembliesResolver
    • IHttpControllerTypeResolver
    • HttpControllerTypeCache
    • IHttpControllerSelector
  3. ServicesContainer

从上节我们知道,在消息管道中,最终在HttpControllerDispatcher的SendAsync方法中会创建IHttpController.

HttpController

我们先看看IHttpController

public interface IHttpController
{
Task<HttpResponseMessage> ExecuteAsync(HttpControllerContext controllerContext, CancellationToken cancellationToken);
}

IHttpController设计很像HttpMessageHandler,只不过参数由HttpRequestMessage变成了HttpControllerContext

public  public class HttpControllerContext
{
HttpControllerContext(HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor)
}

可以看出HttpControllerContext实际就是包装了一下HttpRequestMessage和HttpControllerDescriptor

而HttpControllerDescriptor则是真正创建IHttpController的对象

public class HttpControllerDescriptor
{
public virtual IHttpController CreateController(HttpRequestMessage request)
{
//通过内置的轻量级容器创建
return this.Configuration.Services.GetHttpControllerActivator().Create(request, this, this.ControllerType);
}
}

本节重点就是介绍HttpControllerDescriptor的CreateController方法

创建HttpController

在谈创建HttpController前,我们需要了解HttpControllerDescriptor是如何被创建的

整个创建过程需要经历以下步骤

IAssembliesResolver

首先通过IAssembliesResolver找出符合的程序集

默认的实现是找出当前应用程序域中的程序集

public class DefaultAssembliesResolver : IAssembliesResolver
{
public virtual ICollection<Assembly> GetAssemblies()
{
return (ICollection<Assembly>) ((IEnumerable<Assembly>) AppDomain.CurrentDomain.GetAssemblies()).ToList<Assembly>();
}
}

而在WebHost下,在HttpConfiguration的创建过程中替换为WebHostAssembliesResolver

internal sealed class WebHostAssembliesResolver : IAssembliesResolver
{
ICollection<Assembly> IAssembliesResolver.GetAssemblies()
{
//所有引用的程序集
return (ICollection<Assembly>) BuildManager.GetReferencedAssemblies().OfType<Assembly>().ToList<Assembly>();
}
}

IHttpControllerTypeResolver

在列出可用的Assemblies后,会通过IHttpControllerTypeResolver找出所有的IHttpController

默认实现为DefaultHttpControllerTypeResolver

public class DefaultHttpControllerTypeResolver : IHttpControllerTypeResolver
{
public virtual ICollection<Type> GetControllerTypes(IAssembliesResolver assembliesResolver)
{
foreach (Assembly assembly in (IEnumerable<Assembly>) assembliesResolver.GetAssemblies())
{
typeArray = assembly.GetTypes();
return typeArray.Where<Type>(x => this.IsControllerTypePredicate(x));
}
} internal static bool IsControllerType(Type t)
{
//IsClass IsVisible !IsAbstract IHttpController
if (t != (Type) null && t.IsClass && (t.IsVisible && !t.IsAbstract) && typeof (IHttpController).IsAssignableFrom(t))
return DefaultHttpControllerTypeResolver.HasValidControllerName(t);
return false;
}
}

IHttpControllerSelector

当找到所有HttpController后,IHttpControllerSelector用来选择能够生成HttpControllerDescriptor(当同一个Controller名在不同的命名空间下,这时是WebAPI会抛弃这2个Controller)

默认实现为DefaultHttpControllerSelector

public class DefaultHttpControllerSelector : IHttpControllerSelector
{
//根据请求 选择对应的HttpControllerDescriptor
public virtual HttpControllerDescriptor SelectController(HttpRequestMessage request)
{
string controllerName = this.GetControllerName(request);
HttpControllerDescriptor controllerDescriptor;
if (GetControllerDescriptor(controllerName, out controllerDescriptor))
return controllerDescriptor;
return null;
} //从路由变量controller中获取ControllerName
public virtual string GetControllerName(HttpRequestMessage request)
{
string str;
request.GetRouteData().Values.TryGetValue<string>("controller", out str);
return str;
} public HttpControllerDescriptor GetControllerDescriptor(string controllerName,out HttpControllerDescriptor controllerDescriptor)
{
//去除同名Controller
var keys = ControllerTypeCache.Cache.Where(x=>x.Value.Count == 1).Select(x=>x.Key);
foreach(var key in keys){
//...
return new HttpControllerDescriptor(...);
}
}
}

HttpControllerTypeCache

这里再补充一点:

在DefaultHttpControllerSelector的代码中,我们使用ControllerTypeCache.Cache

实际上由于频繁的反射创建Controller,WebAPI通过缓存Controller避免了不必要的损失.

同时在DefaultHttpControllerSelector代码中也存在相应的缓存措施(文中的代码为了便于理解).

internal sealed class HttpControllerTypeCache
{
public HttpControllerTypeCache(HttpConfiguration configuration)
{
Cache = GetCache(configuration);//该处同样为伪代码
}
internal Dictionary<string, ILookup<string, Type>> Cache {get;}
}

通过上面的流程,我们清楚了HttpControllerDescriptor是如何创建的,同时创建完HttpControllerDescriptor后,调用CreateController即可创建 public virtual IHttpController CreateController(HttpRequestMessage request)

ServicesContainer

在Web API消息管道中,定义了很多接口,对应的也有很多实现类.

WebAPI自定义了一套DI容器ServicesContainer.而这个容器直接挂在HttpConfiguration上

public class HttpConfiguration : IDisposable
{
public ServicesContainer Services { get; internal set; }
public HttpConfiguration(HttpRouteCollection routes)
{
this.Services = (ServicesContainer) new DefaultServices(this);
}
}

在默认的实现DefaultServices 则定义了WebAPI 默认对应的实现

public class DefaultServices : ServicesContainer
{
private readonly Dictionary<Type, object> _defaultServicesSingle = new Dictionary<Type, object>(); public DefaultServices(HttpConfiguration configuration)
{
//...
this.SetSingle<IAssembliesResolver>((IAssembliesResolver) new DefaultAssembliesResolver());
this.SetSingle<IHttpControllerActivator>((IHttpControllerActivator) new DefaultHttpControllerActivator());
this.SetSingle<IHttpControllerSelector>((IHttpControllerSelector) new DefaultHttpControllerSelector(configuration));
this.SetSingle<IHttpControllerTypeResolver>((IHttpControllerTypeResolver) new DefaultHttpControllerTypeResolver());
}
//设置接口实现
private void SetSingle<T>(T instance) where T : class
{
this._defaultServicesSingle[typeof (T)] = (object) instance;
}
//获取接口实现
public override object GetService(Type serviceType)
{
return this._defaultServicesSingle[serviceType];
}
}

备注:

  • 文章中的代码并非完整WebAPI代码,一般是经过自己精简后的.

  • 本篇内容使用MarkDown语法编辑

首发地址:http://neverc.cnblogs.com/p/5952821.html

[Web API] Web API 2 深入系列(3) 激活HttpController的更多相关文章

  1. ASP.NET Web API - ASP.NET MVC 4 系列

           Web API 项目是 Windows 通信接口(Windows Communication Foundation,WCF)团队及其用户激情下的产物,他们想与 HTTP 深度整合.WCF ...

  2. 如何用Baas快速在腾讯云上开发小程序-系列1:搭建API & WEB WebSocket 服务器

    版权声明:本文由贺嘉 原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/221059001487422606 来源:腾云阁 h ...

  3. Java web与web gis学习笔记(二)——百度地图API调用

    系列链接: Java web与web gis学习笔记(一)--Tomcat环境搭建 Java web与web gis学习笔记(二)--百度地图API调用 JavaWeb和WebGIS学习笔记(三)-- ...

  4. HTML5权威指南--Web Storage,本地数据库,本地缓存API,Web Sockets API,Geolocation API(简要学习笔记二)

    1.Web Storage HTML5除了Canvas元素之外,还有一个非常重要的功能那就是客户端本地保存数据的Web Storage功能. 以前都是用cookies保存用户名等简单信息.   但是c ...

  5. 我所理解的RESTful Web API [Web标准篇]

    REST不是一个标准,而是一种软件应用架构风格.基于SOAP的Web服务采用RPC架构,如果说RPC是一种面向操作的架构风格,而REST则是一种面向资源的架构风格.REST是目前业界更为推崇的构建新一 ...

  6. 重构Web Api程序(Api Controller和Entity)续篇

    昨天有写总结<重构Web Api程序(Api Controller和Entity)>http://www.cnblogs.com/insus/p/4350111.html,把一些数据交换的 ...

  7. web api写api接口时返回

    web api写api接口时默认返回的是把你的对象序列化后以XML形式返回,那么怎样才能让其返回为json呢,下面就介绍两种方法: 方法一:(改配置法) 找到Global.asax文件,在Applic ...

  8. Google Maps API Web Services

    原文:Google Maps API Web Services 摘自:https://developers.google.com/maps/documentation/webservices/ Goo ...

  9. asp.net web api 构建api帮助文档

    1 概要 创建ASP.NET Web Api 时模板自带Help Pages框架. 2 问题 1)使用VS创建Web Api项目时,模板将Help Pages框架自动集成到其中,使得Web Api项目 ...

随机推荐

  1. CYQ.Data 批量添加数据性能测试(每秒千、万)

    今天有网友火晋地同学进了CYQ.Data官方群了,他正在折腾了一个各大ORM性能测试的比较的软件,如下图 折腾的种类也不少: 感觉这软件折腾的不错~~~值的期待~~~ 另外,他指出CYQ.Data 在 ...

  2. ArcGIS制作放射状流向地图(Radial Flow Map)

    流向地图火了,因为Facebook的那张著名的友邻图,抑或因为<数据可视化之美>中介绍飞行模式的航线图,总之,流向地图以它特殊的可视化形式,直观地展示事物之间的联系,尤其在展示网络流向.贸 ...

  3. Functional Programming without Lambda - Part 2 Lifting, Functor, Monad

    Lifting Now, let's review map from another perspective. map :: (T -> R) -> [T] -> [R] accep ...

  4. ASP.NET MVC 5 - 验证编辑方法(Edit method)和编辑视图(Edit view)

    在本节中,您将验证电影控制器生成的编辑方法(Edit action methods)和视图.但是首先将修改点代码,使得发布日期属性(ReleaseDate)看上去更好.打开Models \ Movie ...

  5. EF:oracle的number类型映射为C#的boolean类型

    一开始用下面的方法映射, Property(p => p.IsFixed).HasColumnName("IS_FIXED").HasColumnType("num ...

  6. RX(一)

    建议,先去了解观察者模式. 前期工作:引入RX的包 build.gradle脚本里面的 dependencies{ compile 'io.reactivex:rxjava:1.1.0'compile ...

  7. iOS----自定义UIView,绘制一个UIView

    绘制一个UIVIew最灵活的方式就是由它自己完成绘制.实际上你不是绘制一个UIView,你只是子类化了UIView并赋予子类绘制自己的能力.当一个UIVIew需要执行绘图操作的时,drawRect:方 ...

  8. mysql导入导出sql文件

    window下 1.导出整个数据库mysqldump -u 用户名 -p 数据库名 > 导出的文件名mysqldump -u dbuser -p dbname > dbname.sql2. ...

  9. 前端工程师技能之photoshop巧用系列第三篇——切图篇

    × 目录 [1]切图信息 [2]切图步骤 [3]实战 前面的话 前端工程师除了使用photoshop进行测量之外,更重要的是要使用该软件进行切图.本文是photoshop巧用系列的第三篇——切图篇 切 ...

  10. CSS3制作心形头像

    1.功能需求: 最近有一个基于微信开发的Mobile Web项目,是一个活动页面.功能需求:用户使用微信扫描二维码,然后授权使用微信登录,然后读取用户的昵称和头像,然后显示在一个饼图上面.头像需要有一 ...