深入理解asp.net里的HttpModule机制
刚工作的时候看《asp.net深入解析》,第一次知道HttpModule和HttpHandler。当时对我而言,它们不过就是两个新名词而已,仅仅知道工作原理但是理解的不深刻。随着经验的累积,逐渐发现它们对开发的重要性。现在回头再看一遍它们的实现机制,顺便参考一下其他资源做个透彻的读书笔记。
1、asp.net的HTTP请求处理过程
说明:
(1)、客户端浏览器向服务器发出一个http请求,此请求会被inetinfo.exe进程截获,然后转交给aspnet_isapi.dll进程,接着它又通过Http Pipeline的管道,传送给aspnet_wp.exe这个进程,接下来就到了.net framework的HttpRunTime处理中心,处理完毕后就发送给用户浏览器。
(2)、当一个http请求被送入到HttpRuntime之后,这个Http请求会继续被送入到一个被称之为HttpApplication Factory的一个容器当中,而这个容器会给出一个HttpApplication实例来处理传递进来的http请求,而后这个Http请求会依次进入到如下几个容器中:HttpModule --> HttpHandler Factory --> HttpHandler。当系统内部的HttpHandler的ProcessRequest方法处理完毕之后,整个Http Request就被处理完成了,客户端也就得到相应的东东了。
(3)完整的http请求在asp.net framework中的处理流程:
HttpRequest-->inetinfo.exe->ASPNET_ISAPI.DLL-->Http Pipeline-->ASPNET_WP.EXE-->HttpRuntime-->HttpApplication Factory-->HttpApplication-->HttpModule-->HttpHandler Factory-->HttpHandler-->HttpHandler.ProcessRequest()
ps:红色的HttpApplication实例在HttpModule的Init方法中会用到。
(4)如果想在中途截获一个httpRequest并做些自己的处理,就应该在HttpRuntime运行时内部来做到这一点,确切的说是在HttpModule这个容器中来实现。
2、HttpModule工作原理
负责监听HttpRequest,同时对HttpRequest增添或者过滤掉一部分内容。也就是说,当一个HTTP请求到达HttpModule时,整个ASP.NET Framework系统还并没有对这个HTTP请求做任何处理,也就是说此时对于HTTP请求来讲,HttpModule是一个HTTP请求的“必经之路”,所以可以在这个HTTP请求传递到真正的请求处理中心(HttpHandler)之前附加一些需要的信息在这个HTTP请求信息之上,或者针对截获的这个HTTP请求信息作一些额外的工作,或者在某些情况下干脆终止满足一些条件的HTTP请求,从而可以起到一个Filter过滤器的作用。
HttpModule实现了接口IHttpModule,我们可以自定义实现该接口的类,从而取代HttpModule。
asp.net默认的HttpModule如下:
- System.Web.SessionState.SessionStateModule;
- System.Web.Security.WindowsAuthenticationModule;
- System.Web.Security.FormsAuthenticationModule;
- System.Web.Security.PassportAuthenticationModule;
- System.Web.Security.UrlAuthorizationModule;
- System.Web.Security.FileAuthorizationModule;
3、编写自己的HttpModule
要实现HttpModule,必须实现接口IHttpModule。下面是IHttpModule接口分析:
- using System;
- namespace System.Web
- {
- // Summary:
- // Provides module initialization and disposal events to the implementing class.
- public interface IHttpModule
- {
- // Summary:
- // Disposes of the resources (other than memory) used by the module that implements
- // System.Web.IHttpModule.
- // 销毁不再被HttpModule使用的资源
- void Dispose();
- //
- // Summary:
- // Initializes a module and prepares it to handle requests.
- //
- // Parameters:
- // context:
- // An System.Web.HttpApplication that provides access to the methods, properties,
- // and events common to all application objects within an ASP.NET application
- // 初始化一个Module,为捕获HttpRequest做准备
- void Init(HttpApplication context);
- }
- }
下面是自己的HttpModule:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Web; //要在自己的类库中添加System.Web引用
- namespace MyHttpModule
- {
- class MySelfHttpModule : IHttpModule
- {
- public void Dispose() { }
- public void Init(HttpApplication application)
- {
- application.BeginRequest += new EventHandler(Application_BeginRequest);
- application.EndRequest += new EventHandler(Application_EndRequest);
- }
- // 自己要针对一些事情进行处理的两个方法
- private void Application_BeginRequest(object sender, EventArgs e)
- {
- HttpApplication application = sender as HttpApplication;
- HttpContext context = application.Context;
- HttpRequest request = application.Request;
- HttpResponse response = application.Response;
- response.Write("我来自自定义HttpModule中的BeginRequest<br />");
- }
- private void Application_EndRequest(object sender, EventArgs e)
- {
- HttpApplication application = sender as HttpApplication;
- HttpContext context = application.Context;
- HttpRequest request = application.Request;
- HttpResponse response = application.Response;
- response.Write("我来自自定义HttpModule中的EndRequest<br />");
- }
- }
- }
最后在web项目中使用自己的HttpModule:
(1)添加引用
a、不需要在整个web项目添加对类库的引用,只是复制一份到bin目录下即可。
b、default.aspx的cs文件:
- using System;
- using System.Data;
- using System.Configuration;
- using System.Web;
- using System.Web.Security;
- using System.Web.UI;
- using System.Web.UI.WebControls;
- using System.Web.UI.WebControls.WebParts;
- using System.Web.UI.HtmlControls;
- public partial class _Default : System.Web.UI.Page
- {
- protected void Page_Load(object sender, EventArgs e)
- {
- Response.Write("<br/>来自Default.aspx页面<br/>");
- }
- }
(2)web.config的节点配置
在Web.config的system.web标签中添加:
<httpModules>
<add name="MyHttpModuleTest" type="MyHttpModule.MyHttpModule,MyHttpModule"></add>
</httpModules>
a、name可以随意指定,没有影响。
b、type有两个参数,第一个表示具体哪个类,第二个表示是哪个dll。
最后运行web项目,你会看到每个页面都会Response.Write两句话出来。
4、HttpModule内部事件机制和生命周期
HttpModule对HttpApplication实例进行处理,而HttpApplication有很多事件(对应不同的生命期),这样就衍生出HttpModule内部事件机制和生命周期。
(1)HttpModule的事件
说明:
a、BenginRequest和EndRequest分别是HttpModule容器最开始的和最后的事件;
b、EndRequest之后还会触发PreSendRequestHeaders事件和PreSendRequestContent事件,这不是在HttpModule外的两个事件,表示HttpModule结束,即将开始向Client发送数据。
(2)、验证HttpModule生命周期
与HttpHandler的交互:
说明:
a、HttpModule容器会将HttpRequest传递到HttpHandler容器,这个时间点是ResolveRequestCache事件
b、HttpModule容器会建立HttpHandler实例作为入口——Session从此生效
c、触发AcquireRequestState事件以及PreRequestHandlerExecute事件
d、HttpModule容器便将对HttpRequest的控制权转让给HttpHandler容器
e、HttpHandler容器处理HttpRequest——使用自身的ProcessRequest方法,将对其控制权又还给HttpModule容器——之后Session失效。
验证生命周期代码:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Web;
- namespace MyHttpModule
- {
- public class ValidaterHttpModuleEvents : IHttpModule
- {
- public void Dispose()
- { }
- /// <summary>
- /// 验证HttpModule事件机制
- /// </summary>
- /// <param name="application"></param>
- public void Init(HttpApplication application)
- {
- application.BeginRequest += new EventHandler(application_BeginRequest);
- application.EndRequest += new EventHandler(application_EndRequest);
- application.AcquireRequestState += new EventHandler(application_AcquireRequestState);
- application.AuthenticateRequest += new EventHandler(application_AuthenticateRequest);
- application.AuthorizeRequest += new EventHandler(application_AuthorizeRequest);
- application.PreRequestHandlerExecute += new EventHandler(application_PreRequestHandlerExecute);
- application.PostRequestHandlerExecute += new EventHandler(application_PostRequestHandlerExecute);
- application.ReleaseRequestState += new EventHandler(application_ReleaseRequestState);
- application.ResolveRequestCache += new EventHandler(application_ResolveRequestCache);
- application.PreSendRequestHeaders += new EventHandler(application_PreSendRequestHeaders);
- application.PreSendRequestContent += new EventHandler(application_PreSendRequestContent);
- }
- private void application_BeginRequest(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_BeginRequest<br/>");
- }
- private void application_EndRequest(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_EndRequest<br/>");
- }
- private void application_PreRequestHandlerExecute(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_PreRequestHandlerExecute<br/>");
- }
- private void application_PostRequestHandlerExecute(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_PostRequestHandlerExecute<br/>");
- }
- private void application_ReleaseRequestState(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_ReleaseRequestState<br/>");
- }
- private void application_AcquireRequestState(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_AcquireRequestState<br/>");
- }
- private void application_PreSendRequestContent(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_PreSendRequestContent<br/>");
- }
- private void application_PreSendRequestHeaders(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_PreSendRequestHeaders<br/>");
- }
- private void application_ResolveRequestCache(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_ResolveRequestCache<br/>");
- }
- private void application_AuthorizeRequest(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_AuthorizeRequest<br/>");
- }
- private void application_AuthenticateRequest(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("application_AuthenticateRequest<br/>");
- }
- }
- }
注:修改配置文件中的httpModules节点中的type对应的类名即可.
5、其他
(1)、加载两个或多个自定义的HttpModule
这里以两个自定义HttpModule举例.
修改配置文件中的httpModules节点:
<add name="HttpModule1" type="MyHttpModule.HttpModule1,MyHttpModule"/>
<add name="HttpModule2" type="MyHttpModule.HttpModule2,MyHttpModule"/>
HttpModule1和HttpModule2模仿ValidaterHttpModuleEvents编写(除了类名改变外,事件和方法不变),不贴代码了。运行结果如下:
分析:
a、从运行结果可以看到,在web.config文件中引入自定义HttpModule的顺序就决定了多个自定义HttpModule在处理一个HTTP请求的接管顺序;
b、系统默认那几个HttpModule是最先被ASP.NET Framework所加载上去的,对外部是透明的。
(2)、利用HttpModule实现终止此次HttpRequest请求
在BeginRequest事件中,使用HttpApplication.CompleteRequest()方法可以实现当满足一定条件时终止此次HttpRequest请求。看代码:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Web;
- namespace MyHttpModule
- {
- public class CompleteRequestHttpModule : IHttpModule
- {
- public void Dispose() { }
- public void Init(HttpApplication application)
- {
- application.BeginRequest += new EventHandler(Application_BeginRequest);
- }
- private void Application_BeginRequest(object sender, EventArgs e)
- {
- HttpApplication application = (HttpApplication)sender;
- application.Context.Response.Write("请求<br/>");
- application.CompleteRequest(); //在BeginRequest事件中,使用HttpApplication.CompleteRequest()方法实现
- application.Context.Response.Write("请求被终止");
- }
- }
- }
说明:
a、对于一个HttpModule,在BeginRquest中终止,但是仍然会调用EndRequest事件,以及PreSendRequestHeaders事件和PreSendRequestContent事件。也可以说是直接跳转到EndRequest事件,而不会调用这期间的事件
b、如果有两个HttpModule,在第一个HttpModule的BeginRequest中终止,仅仅不会调用第二个HttpModule的BeginRequest,但仍然会调用两个EndRequest事件,以及PreSendRequestHeaders事件和PreSendRequestContent事件。看下面的图示:
深入理解asp.net里的HttpModule机制的更多相关文章
- 深入理解ASP.NET的内部运行机制(转)
WebForms和WebServices作为.NET平台构建Web程序的两大利器,以其开发简单.易于部署的特点得到了广泛的应用,但殊不知微软公司在背后为我们做了大量的基础性工作,以至于我们开发人员只需 ...
- ASP.NET MVC的运行机制--url的全局分析
全局 首先我们来看一副图片 首先,用户通过Web浏览器向服务器发送一条url请求,这里请求的url不再是xxx.aspx格式,而是http://HostName/ControllerNam ...
- ASP.NET三剑客 HttpApplication HttpModule HttpHandler 解析
我们都知道,ASP.Net运行时环境中处理请求是通过一系列对象来完成的,包含HttpApplication,HttpModule, HttpHandler.之所以将这三个对象称之为ASP.NET三剑客 ...
- 理解asp.net中DropDownList编辑数据源,绑定数据库数据。
一.理解asp.net绑定数据库 终于学习到了连接数据库部分的内容,杨中科老师视频看起来挺轻松的,如果是高清版就更ok了. 我发现我学习新的编程语言会有一个特点,都想要赶紧学习数据库,数据就是一切,有 ...
- 《深入理解mybatis原理》 Mybatis初始化机制具体解释
对于不论什么框架而言.在使用前都要进行一系列的初始化,MyBatis也不例外. 本章将通过下面几点具体介绍MyBatis的初始化过程. 1.MyBatis的初始化做了什么 2. MyBatis基于XM ...
- 深入理解ASP.NET MVC(5)
系列目录 回顾 系列的前4节深入剖析了ASP.NET URL路由机制,以及MVC在此基础上是如何实现Areas机制的,同时涉及到inbound和outbound很多细节部分.第2节中提到MvcRout ...
- 简述ASP.NET的页面运行机制
在深入学习ASP.NET之前,首先需要先了解一下ASP.NET的页面运行机制: 浏览以下内容需要对ASP.NET管道有一定的了解,附上很不错的几个链接: 选择HttpHandler还是HttpModu ...
- 深入理解ASP.NET MVC(目录)
学ASP.NET MVC2有一段时间了,也针对性的做了个练习.感觉这个框架还是不错的,所以决定要深入系统的学习一下.看到这样一本书: 作者博客:http://blog.stevensanderson. ...
- 使用生活实例理解Asp.net运行时
学习编程语言,掌握面向对象的编程思想尤为重要,一旦理解了面向对象的这种概念,那么好些地方拿到生活中去理解,就容易的多了.书本上的枯燥干涩的语言,对于好多人来说,即难懂,更难长时间牢牢记得.但是编程语言 ...
随机推荐
- 【原】iOS学习之PCH文件
1. PCH文件概述 PCH文件是一种预编译头文件(一般扩展名为.PCH),是把一个工程中较稳定的代码预先编译好放在一个文件(.PCH)里.这些预先编译好的代码可以是任何的C/C++代码--甚至可以是 ...
- soapui中文操作手册(六)----创建REST Testing
首先,通过选择文件菜单中的“新建REST项目”选项创建从文件菜单中一个新的REST项目: 指定服务端点场下谷歌地图API网址: http://maps.googleapis.com/maps/api/ ...
- android 第三方 图表
1.XCL-Charts 直接利用Canvas api画出图形,各有好处. XCL-Chart尽量把图的绘制逻辑封装在类中,而把绘制相关的各 个元素开放出来,目的是在保证开发效率的同时,给程序员足够多 ...
- ACM: HDU 1285 确定比赛名次 - 拓扑排序
HDU 1285 确定比赛名次 Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u De ...
- phpunit测试成功 phpunit测试实践代码
16:12 2015/12/8phpunit测试成功,代码写在www目录下,以类名命名代码文件,我的文件名为 ArrayTest.php,类名为ArrayTest,内部写了简单的测试代码:<?p ...
- 【BZOJ】2729: [HNOI2012]排队
题意 \(n\)个男生\(m\)个女生\(2\)个老师排列,任意两个女生不能相邻,两个老师也不能相邻,每个人都不同.问有多少种排法.(\(n, m \le 2000\)) 分析 组合乱搞. 题解 先放 ...
- 【bzoj2049】[Sdoi2008]Cave 洞穴勘测 link-cut-tree
2016-05-30 11:04:51 学习了link-cut-tree 二中神犇封禹的讲义感觉讲的超级清晰易懂啊(没有的可以q窝 算是模板吧 #include<bits/stdc++.h&g ...
- db2代理和优化
DB2 的代理 (agent) 是位于 DB2 服务器中的服务于应用程序请求的一些进程或线程.当有外部应用程序连接至 DB2 实例提出访问请求时,DB2 的代理就会被激活去应答这些请求.一般 DB2 ...
- C#_生成HTML
#region 生成静态页 /// <summary> /// 生成静态页 /// </summary> /// <param name="URL"& ...
- 用简单直白的方式讲解A星寻路算法原理
很多游戏特别是rts,rpg类游戏,都需要用到寻路.寻路算法有深度优先搜索(DFS),广度优先搜索(BFS),A星算法等,而A星算法是一种具备启发性策略的算法,效率是几种算法中最高的,因此也成为游戏中 ...