我们知道项目中创建的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. iOS.Operation-on-ZipFile

    Operation on ZipFile Reference 在Mac OS X和iOS中操作.zip文件(例如创建zip文件, 从zip文件中抽取数据): 1. http://stackoverfl ...

  2. sql重复数据的过滤问题

    有重复数据主要有一下几种情况: 1.存在两条完全相同的纪录 这是最简单的一种情况,用关键字distinct就可以去掉 example: select distinct * from table(表名) ...

  3. delphi新手到高手的工具--castalia

    castalia翻译是 神泉 ,是delphi的一个优秀第三方工具.其重构功能尤为突出.代码实时编译提示也很棒. 自卑delphi开发工具没有eclipse那么强大的提示?有castalia为你提升信 ...

  4. mysqli_query数据库有数据,查不出来

    MySQLDB.class.php <?php /** * 数据库操作工具类 */ class MySQLDB { // 定义相关属性 private $host;// 主机地址 private ...

  5. Maximum Average Subarray I LT643

    Given an array consisting of n integers, find the contiguous subarray of given length k that has the ...

  6. 用 Python 写 Robot Framework 测试

    Robot Framework 框架是基于 Python 语言开发的,所以,它本质上是 Python 的一个库. 1.你懂 Python 语言. 2.又想使用 Robot Framework 测试框架 ...

  7. 【Linux】CentOS 7.4 安装 MySQL 8.0.12 解压版

    安装环境/工具 1.Linux(CentOS 7.4版) 2.mysql-8.0.12-el7-x86_64.tar.gz 安装步骤 参考:https://dev.mysql.com/doc/refm ...

  8. mybatis学习 十六 auto_mapping实现连表查询

    只能使用多表联合查询方式. 要求:查询出的列别和属性名相同. 点字符  "."  在 SQL 是关键字符,两侧添加反单引号(Tab键上的一个字符) <select id=&q ...

  9. 【转】CentOS 7 安装配置 NFS

    环境 nps 192.168.1.97 client 192.168.1.98 一.yum 安装 yum -y install nfs-utils rpcbind nfs 的配置文件 /etc/exp ...

  10. 2018.11.16 bzoj4827: [Hnoi2017]礼物(ntt)

    传送门 nttnttntt 入门题. 考虑展开要求的式子∑i=0n−1(xi−yi−c)2\sum_{i=0}^{n-1}(x_i-y_i-c)^2∑i=0n−1​(xi​−yi​−c)2 => ...