[转载]ASP.NET底层机制 (上) HttpModule
在IIS的应用程序映射中,IIS会将对asp和asa文件的请求转交给asp.dll这个ISAPI来处理
解释型处理,每次请求都会重新解释一次,不适用于大流量请求
2.ASP.NET的HTTP请求处理过程
3.在HttpRuntime中请求处理过程
HttpRequest请求:
进入HttpRumtime——通过HttpApplicationFactory,建立HttpApplication实例——进入HttpModule——通过HttpHandlerFactory,建立HttpHandler实例
*这个HttpApplication实例在HttpModule的Init方法中会用到
4.HttpModule工作原理
负责监听HttpRequest,同时对HttpRequest增添或者过滤掉一部分内容。
HttpModule实现了接口IHttpModule,我们可以自定义实现该接口的类,从而取代HttpModule。
ASP.NET默认的HttpModule如下:
System.Web.Security.WindowsAuthenticationModule;
System.Web.Security.FormsAuthenticationModule;
System.Web.Security.PassportAuthenticationModule;
System.Web.Security.UrlAuthorizationModule;
System.Web.Security.FileAuthorizationModule;
IHttpModule接口分析:
{
// 销毁不再被HttpModule使用的资源
void Dispose();
//初始化一个Module,为捕获HttpRequest做准备
void Init(HttpApplication context);
}
编写自己的HttpModule:
using System;
using System.Web;
namespace ClassLibraryModule
{
public class MyHttpModule : IHttpModule
{
public void Init(HttpApplication context)
{
context.BeginRequest += new EventHandler(this.Application_BeginRequest);
context.EndRequest += new EventHandler(this.Application_EndRequest);
}
public void Dispose() { }
//自己要处理私事的两个方法
public void Application_BeginRequest(Object sender, EventArgs e)
{
HttpApplication application = (HttpApplication)sender;
HttpContext context = application.Context;
HttpResponse response = application.Response;
HttpRequest request = application.Request;
response.Write("来自Application_BeginRequest");
}
public void Application_EndRequest(Object sender, EventArgs e)
{
HttpApplication application = (HttpApplication)sender;
HttpContext context = application.Context;
HttpResponse response = application.Response;
HttpRequest request = application.Request;
response.Write("来自Application_EndRequest");
}
}
}
在Web项目中添加这个类库的引用,同时在Web.config的system.web标签中添加:
<add name="Test" type="ClassLibraryModule.MyHttpModule,ClassLibraryModule"></add>
</httpModules>
name可以随意指定,没有影响。
type有两个参数,第一个表示具体哪个类,第二个表示是哪个dll
于是该站点下的每个页面都会Response.Write两句话——这适合做广告,只要替换成javascript即可
5.HttpModule内部事件机制
HttpApplication实例有很多事件,BenginRequest和EndRequest分别是HttpModule容器最开始的和最后的事件
注意,EndRequest之后还会触发PreSendRequestHeaders事件和PreSendRequestContent事件,这不是在HttpModule外的两个事件,表示HttpModule结束,即将开始向Client发送数据。
HttpModule容器与HttpHandler容器的交互:
HttpModule容器会将HttpRequest传递到HttpHandler容器,这个时间点是ResolveRequestCache事件。
HttpModule容器会建立HttpHandler实例作为入口——Session从此生效
触发AcquireRequestState事件以及PreRequestHandlerExecute事件,
HttpModule容器便将对HttpRequest的控制权转让给HttpHandler容器。
HttpHandler容器处理HttpRequest——使用自身的ProcessRequest方法,将对其控制权又还给HttpModule容器——之后Session失效
可以同时加载两个HttpModule,
<add name="Test1" type="ClassLibraryModule.MyHttpModule1,ClassLibraryModule1"></add>
<add name="Test2" type="ClassLibraryModule.MyHttpModule2,ClassLibraryModule2"></add>
</httpModules>
这时,根据add标签的先后,依次执行:
Test1.BeginRequest
Test2.BeginRequest
.....
Test1.EndRequest
Test2.EndRequest
利用HttpModule实现当满足一定条件时终止此次HttpRequest:
在BeginRequest事件中,使用HttpApplication.CompleteRequest()方法
{
HttpApplication application = (HttpApplication)sender;
HttpContext context = application.Context;
application.CompleteRequest();
context.Response.StatusCode = 500;
context.Response.StatusDescription = "Internal Server Error";
}
如果有两个HttpModule,在第一个Module的BeginRequest中终止,仅仅不会调用第二个Module的BeginRequest, 但仍然会调用两个EndRequest事件,以及PreSendRequestHeaders事件和PreSendRequestContent事件。
以上两句话,可以用下图来表示: