我们知道项目中创建的Controller,如ProductController都继承自ApiController抽象类,其又实现了接口IDisposable,所以,框架中自动调用Dispose方法来释放其资源。在代表请求的HttpRequestMessage属性字典中,有个Key,“Ms_DisposableRequestResources” ,其值是类型List<IDisposable>,用来存放待释放的资源,回顾下ApiController抽象类的方法ExecuteAsync里的代码片段。 

  Request.RegisterForDispose(this)是主要代码,我们看下HttpRequestMessage的相关扩展方法,RegisterForDispose注册销毁资源,DisposeRequestResources释放所有资源。

public static class HttpRequestMessageExtensions
{
//注册一个待销毁的资源列表到请求的属性字典中
public static void RegisterForDispose(this HttpRequestMessage request, IDisposable resource)
{
if (request == null)
{
throw Error.ArgumentNull("request");
} if (resource == null)
{
return;
} List<IDisposable> trackedResources = GetRegisteredResourcesForDispose(request); trackedResources.Add(resource);
} //一次性注册多个待销毁的资源列表到请求的属性字典中
public static void RegisterForDispose(this HttpRequestMessage request, IEnumerable<IDisposable> resources)
{
if (request == null)
{
throw Error.ArgumentNull("request");
} if (resources == null)
{
throw Error.ArgumentNull("resources");
} List<IDisposable> trackedResources = GetRegisteredResourcesForDispose(request); foreach (IDisposable resource in resources)
{
if (resource != null)
{
trackedResources.Add(resource);
}
}
} //释放资源列表(RegisterForDispose方法注册)里的所有资源
public static void DisposeRequestResources(this HttpRequestMessage request)
{
if (request == null)
{
throw Error.ArgumentNull("request");
} List<IDisposable> resourcesToDispose;
if (request.Properties.TryGetValue(HttpPropertyKeys.DisposableRequestResourcesKey, out resourcesToDispose))
{
foreach (IDisposable resource in resourcesToDispose)
{
try
{
resource.Dispose();
}
catch
{
// ignore exceptions
}
}
//清空列表
resourcesToDispose.Clear();
}
} //从当前请求中获取所有待销毁的资源的列表
public static IEnumerable<IDisposable> GetResourcesForDisposal(this HttpRequestMessage request)
{
if (request == null)
{
throw Error.ArgumentNull("request");
} return GetRegisteredResourcesForDispose(request);
} //从当前请求中获取所有待销毁的资源的列表
private static List<IDisposable> GetRegisteredResourcesForDispose(HttpRequestMessage request)
{
List<IDisposable> registeredResourcesForDispose;
//从属性字典中,获取指定Key的值
if (!request.Properties.TryGetValue(HttpPropertyKeys.DisposableRequestResourcesKey, out registeredResourcesForDispose))
{
registeredResourcesForDispose = new List<IDisposable>();
request.Properties[HttpPropertyKeys.DisposableRequestResourcesKey] = registeredResourcesForDispose;
}
return registeredResourcesForDispose;
}
}

一、WebHost 模式下的资源销毁

  ASP.NET Web API用于“处理请求、回复响应”的 HttpMessageHandler管道是由 HttpControllerHandler创建的,后者根据当前HTTP上下文创建一个表示当前请求的HttpRequestMessage对象并传入这个管道进行处理 。在整个管道完成对请求的处理并最终对请求予以响应之后 ,HttpControllerHandler会负责完成资源释放有关的工作

工作。回顾下HttpControllerHandler,方法的代码:

public  class HttpControllerHandler:HttpTaskAsyncHandler
{
//省略其他成员 public HttpControllerHandler(RouteData routeData)
: this(routeData, GlobalConfiguration.DefaultServer)
{
} public override Task ProcessRequestAsync(HttpContext context)
{
return ProcessRequestAsyncCore(new HttpContextWrapper(context));
} internal async Task ProcessRequestAsyncCore(HttpContextBase contextBase)
{
HttpRequestMessage request = contextBase.GetHttpRequestMessage() ?? ConvertRequest(contextBase); // Add route data
request.SetRouteData(_routeData);
CancellationToken cancellationToken = contextBase.Response.GetClientDisconnectedTokenWhenFixed();
HttpResponseMessage response = null; try
{
//创建消息处理管道,并执行
response = await _server.SendAsync(request, cancellationToken);
//异步执行完成
await CopyResponseAsync(contextBase, request, response, _exceptionLogger.Value, _exceptionHandler.Value,
cancellationToken);
}
catch (OperationCanceledException)
{ contextBase.Request.Abort();
}
finally
{
//1.调用方法DisposeRequestResources释放属性字典中的待释放资源
         request.DisposeRequestResources();
         //2.释放请求消息
        request.Dispose();
         //3.释放响应消息
if (response != null)
{
response.Dispose();
}
}
}
}

  从代码可知道,创建管道并执行完成后,在finally代码块里释放了三种资源,

  • 调用方法DisposeRequestResources释放属性字典中的待释放资源
  • 请求消息
  • 响应消息

二、Self Host模式下的资源销毁

  Self Host模式下的请求的监听、接收和响应是通过HttpBinding创建的信道栈来完成的,该信道栈处理的消息类型为HttpMessage,具体代表请求消息和响应消息的HttpMesmge分别是对HttpRequestMessage和Message对象的封装。WCF中表示消息的Message本身就是一个需要最终被释放的对象,在针对它的处理结束之后会调用其Close或 者 Dispose方法对它进行资源释放的工作。再看下代码段

internal sealed class HttpMessage : Message
{
//省略其他成员
protected override void OnClose()
{
base.Close();
if (_request != null)
{
//1.调用DisposeRequestResources,释放属性字典中注册的资源
_request.DisposeRequestResources();
//2.释放请求消息
_request.Dispose();
_request = null;
}
//3.释放响应消息
if (_response != null)
{
_response.Dispose();
_response = null;
}
}
}
 internal sealed class Message : IDisposable
{
protected virtual void OnClose()
{
} public void Close()
{
if (this.state != MessageState.Closed)
{
this.state = MessageState.Closed;
//调用OnClose()
this.OnClose();
if (!System.ServiceModel.DiagnosticUtility.ShouldTraceVerbose)
return;
TraceUtility.TraceEvent(TraceEventType.Verbose, , System.ServiceModel.SR.GetString("TraceCodeMessageClosed"), this);
}
else
{
if (!System.ServiceModel.DiagnosticUtility.ShouldTraceVerbose)
return;
TraceUtility.TraceEvent(TraceEventType.Verbose, , System.ServiceModel.SR.GetString("TraceCodeMessageClosedAgain"), this);
}
}
//Dispose模式用法
void IDisposable.Dispose()
{
//调用Close()
this.Close();
}
}

  从代码可以知道,在HttpMessage的Dispose方法里,会调用代码也是释放三种资源

  • 调用方法DisposeRequestResources释放属性字典中的待释放资源
  • 请求消息
  • 响应消息

ASP.NET Web API 框架研究 Controller实例的销毁的更多相关文章

  1. ASP.NET Web API 框架研究 Controller创建 HttpController介绍

    对请求进行路由解析以及消息处理管道进行处理后,最后可以从HttpRequestMessage对象的属性字典中获取解析的路由数据,后边我们就可以根据其进行HttpController的创建,从前边几篇可 ...

  2. ASP.NET Web API 框架研究 Controller创建 HttpController 类型解析 选择 创建

    上一篇介绍了HttpController的一些细节,接下来说下HttpController 类型解析.选择和创建.生产HttpController实例的生产线如下图: 一.涉及的类及源码分析 涉及的类 ...

  3. ASP.NET Web API 框架研究 Controller创建过程与消息处理管道

    现在我们从代码角度来看下,从消息处理管道末尾是怎么创建出Controller实例的.消息处理管道末端是一个叫HttpRoutingDispatcher的处理器,其内部完成路由后 ,会把消息派送给其内部 ...

  4. ASP.NET Web API 框架研究 ASP.NET Web API 路由

    ASP.NET Web API 核心框架是一个独立的.抽象的消息处理管道,ASP.NET Web API有自己独立的路由系统,是消息处理管道的组成部分,其与ASP.NET路由系统有类似的设计,都能找到 ...

  5. ASP.NET Web API 框架研究 Action方法介绍

    在根据请求解析出匹配的Controller类型并创建实例后,要在该Controller类型中的众多Action方法中选择与请求匹配的那一个,并执行,然后返回响应. Action方法,其元数据,主要包括 ...

  6. ASP.NET Web API 框架研究 核心的消息处理管道

    ASP.NET Web API 的核心框架是一个由一组HttpMessageHandler有序组成的双工消息处理管道:寄宿监听到请求接受后,把消息传入该管道经过所有HttpMessageHandler ...

  7. ASP.NET Web API 框架研究 IoC容器 DependencyResolver

    一.概念 1.IoC(Inversion of Control),控制反转 即将依赖对象的创建和维护交给一个外部容器来负责,而不是应用本身.如,在类型A中需要使用类型B的实例,而B的实例的创建不是由A ...

  8. ASP.NET Web API 框架研究 服务容器 ServicesContainer

    ServicesContainer是一个服务的容器,可以理解为—个轻量级的IoC容器,其维护着一个服务接口类型与服务实例之间的映射关系,可以根据服务接口类型获取对应的服务实例.构成ASP.NET We ...

  9. ASP.NET Web API 框架研究 Self Host模式下的消息处理管道

    Self Host模式下的ASP.NET Web API与WCF非常相似,都可以寄宿在任意类型的托管应用程序中,宿主可以是Windows Form .WPF.控制台应用以及Windows Servic ...

随机推荐

  1. centos vncviewer

    CentOS6.5 安装vncserver实现图形化访问   一. 安装gnome图形化桌面 #yum groupinstall -y "X Window System" #yum ...

  2. iOS8 UIAlertView键盘闪一下的问题

    if (SYSTEM_VERSION >= 8.0) { UIAlertController *alertCtrl = [UIAlertController alertControllerWit ...

  3. 关于python的字符编码

    理论特别多,金角大王讲的非常细致和深入浅出. 我来个简短的总结: python2的编码:默认是ascii,可以改变成gbk,utf-8等,但是用什么编码写的,就存储成什么编码.如果搬到linux,默认 ...

  4. kafka的advertised.host.name参数 外网访问配置

    kafka的server.properties文件 ```host.name```开始只绑定在了内部IP上,对外网卡无法访问. 把值设置为空的话会kafka监听端口在所有的网卡上绑定.但是在外网访问时 ...

  5. [FreeMind] 绘制思维时遇到的常见问题解决办法

    如何改变节点的摆放方向? 如果是新建节点,选择要放置节点的那一侧,按enter键,或者鼠标右键,插入平行节点即可. 如果是已经建好的节点,可以用ctrl+x, ctrl+v粘贴到另一边,或者选中子节点 ...

  6. No goals have been specified for this build. You must specify a valid lifecycle phase or a goal in the format <plugin-prefix>:<goal> or <plugin-group-id>:<plugin-artifact-id>[:<plugin-version>]:<goal

    No goals have been specified for this build. You must specify a valid lifecycle phase or a goal in t ...

  7. The Django Book第七章(表单)

    从Request对象中获取数据 在views函数里.我们常常写到 from django.http import HttpResponse def hello(request): return Htt ...

  8. Mac OS X下安装Python的MySQLdb模块【终结版】

    1.下载源文件: https://pypi.org/project/MySQL-python/ 2.cd 到源文件所在目录: 3.在目录下使用 python setup.py install 命令安装 ...

  9. Keras框架下使用CNN进行CIFAR-10的识别测试

    有手册,然后代码不知道看一下:https://keras-cn.readthedocs.io/en/latest/ 首先是下载数据集,下载太慢了就从网盘上下载: 链接:https://pan.baid ...

  10. Tomcat架构解析(四)-----Coyote、HTTP、AJP、HTTP2等协议

    Connector是Tomcat中非常重要的一个组成部分,说白了,就是如何从客户端获取到相应的请求信息.这部分主要包括的难点有这样几个部分: 1.客户端与服务端的协议 客户端与服务端的协议是多种多样的 ...