说明:HttpHandler是一个HTTP请求的真正处理中心。在HttpHandler容器中,ASP.NET Framework才调用HttpHandler的ProcessRequest成员方法来对这个HTTP请求进行真正的处理,真正地对客户端请求的服务器页面做出编译和执行,并将处理过后的信息附加在HTTP请求信息流中再次返回到HttpModule中。
以一个aspx页面为例,正是在HttpHandler这里,一个aspx页面才被系统处理解析,并将处理完成的结果继续经由HttpModule传递下去,直至到达客户端。当然,对于aspx页面,ASP.NET Framework在默认情况下是交给System.Web.UI.PageHandlerFactory这个HttpHandlerFactory来处理的。当一个HTTP请求到达这个HttpHandlerFactory时,HttpHandlerFactory会提供出一个HttpHandler容器,交由这个HttpHandler容器来处理这个HTTP请求。 一个HTTP请求都是最终交给一个HttpHandler容器中的ProcessRequest方法来处理的。
(1)实现HttpHandler,必须继承自IHttpHandler接口。下面是这个接口的定义:
using System;
namespace System.Web
{public interface IHttpHandler{// 其他Request是否可以使用IHttpHandler
bool IsReusable { get; }// 处理HttpRequest
void ProcessRequest(HttpContext context);
}}
(2)自定义HttpHandler
新建一个网站,default.aspx页面:default.aspx.cs:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
namespace WebApplication1
{public partial class _Default : System.Web.UI.Page{protected void Page_Load(object sender, EventArgs e){Response.Write("<br/>来自Default.aspx页面<br/>");
}}}
新添一个类库MyHandler,添加一个类如下:
using System;
using System.Web;
using System.Web.SessionState;
namespace WebApplication1
{public class MyTestHandler : IHttpHandler, IRequiresSessionState{public bool IsReusable { get { return true; } }public void ProcessRequest(HttpContext context){context.Response.Write("<h3><b>This is a HttpHandler Test</b></h3>");
context.Session["Test"] = "<h3><span style=\"color:blue;\">在HttpHandler容器中调用Session</span></h3>";context.Response.Write(context.Session["Test"]);
}}}
(3)配置文件
在web.config文件的system.web节点下,添加:
<add verb="*" path="*.aspx" type="WebApplication1.MyTestHandler, WebApplication1" />
(4)注意
<1>、.NET为asp.net提供了很多系统默认HttpHandler类,用来适应不同类型的HttpRequest。比如aspx,在machine.config中是这样定义的:
<add verb="*" path="*.aspx" type="System.Web.UI.PageHandlerFactory"/>
这就说明遇到aspx的Request请求,asp.net会将其交给System.Web.UI.PageHandlerFactory的HttpHandler类来处理。
<2>、如果自己定义了新的HttpHandler,而且在web.config中指定,则系统只会使用这个新的HttpHandler,而不再使用原先默认的或者指定的.
ASP.NET Framework实际不直接将相关的页面资源HTTP请求定位到一个其内部默认的IHttpHandler容器之上,而定位到了其内部默认的IHttpHandler工厂上。IHttpHandler工厂的作用是对IHttpHandler容器进行调度和管理,这样做的优点是大大增强了系统的负荷性,提升了效率。
(1)IHttpHandlerFactory接口
IHttpHandlerFactory接口包含两个方法:GetHandler方法返回实现IHttpHandler接口的类的实例,ReleaseHandler方法使工厂可以重用现有的处理程序实例。
using System;
using System.Web;
namespace System.Web.UI
{public class PageHandlerFactory : System.Web.IHttpHandlerFactory2, IHttpHandlerFactory{protected internal PageHandlerFactory();public virtual IHttpHandler GetHandler(HttpContext context, string requestType, string virtualPath, string path);public virtual void ReleaseHandler(IHttpHandler handler);}}
(2) 实现一个简单的HttpHandler工厂
类库新添一个文件MyHandlerFactor.cs:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
namespace WebApplication1
{public class MyHandlerFactory : IHttpHandlerFactory{public IHttpHandler GetHandler(HttpContext context, string requestType, string url, string pathTranslated){string fileName = url.Substring(url.LastIndexOf("/") + 1);string objName = fileName.Substring(0, fileName.IndexOf("."));string className = "WebApplication1." + objName;object objHandler = null;try
{// 采用动态反射机制创建相应的IHttpHandler实现类。
objHandler = Activator.CreateInstance(Type.GetType(className));context.Response.Write(className);}catch (Exception e)
{throw new HttpException("工厂不能为类型" + objName + "创建实例。", e);}return (IHttpHandler)objHandler;
}public void ReleaseHandler(IHttpHandler handler){}}public class Handler1 : IHttpHandler{public bool IsReusable { get { return true; } }public void ProcessRequest(HttpContext context){context.Response.Write("<br/>这是来自于MyHandlerFactory里的处理.<br/>");
context.Response.Write("<h3>来自Handler1的信息.</h3>");
}}public class Handler2 : IHttpHandler{public bool IsReusable { get { return true; } }public void ProcessRequest(HttpContext context){context.Response.Write("<br/>这是来自于MyHandlerFactory里的处理.<br/>");
context.Response.Write("<h3>来自Handler2的信息.</h3>");
}}}
<add verb="*" path="Handler1.aspx" type="WebApplication1.MyHandlerFactory,WebApplication1"/><add verb="*" path="Handler2.aspx" type="WebApplication1.MyHandlerFactory,WebApplication1"/>
到这里,针对Handler1.aspx和Handler2.aspx两个页面的http请求我们就通过HttpHandler工厂处理好了。
主要有两点:
(1)先后次序.先IHttpModule,后IHttpHandler,IHttpHandler处理结束后再交给IHttpModule;
(2)对请求的处理上:
IHttpModule是属于大小通吃类型,无论客户端请求的是什么文件,都会调用到它;例如aspx,html,rar的请求;
IHttpHandler则属于挑食类型,只有asp.net注册过的文件类型(例如aspx,ascx,asmx等等)才会轮到调用它。