写在前面
从上家公司离职,在家休息,闲着无聊,觉得还是有必要将IHttpHanlder的内容,做一个总结。发现在写demo的过程中,总觉得有点生疏了,项目中很少使用自定义的类来实现该接口。当然,一般处理程序除外。
IHttpHandler
在IHttpModule这篇文章中,有一个asp.net生命周期的简图。忘记的可以,翻到那篇文章,再熟悉一下。
当一个HTTP请求到达HttpModule时,整个ASP.NET Framework系统还并没有对这个HTTP请求做任何处理,也就是说此时对于HTTP请求来讲,HttpModule是一个HTTP请求的“必经之路”,所以可以在这个HTTP请求传递到真正的请求处理中心(HttpHandler)之前附加一些需要的信息在这个HTTP请求信息之上,或者针对截获的这个HTTP请求信息作一些额外的工作,或者在某些情况下干脆终止满足一些条件的HTTP请求,从而可以起到一个Filter过滤器的作用。
HttpHandler是一个HTTP请求的真正处理中心,也正是在这个HttpHandler容器中,ASP.NET Framework才真正地对客户端请求的服务器页面做出编译和执行,并将处理过后的信息附加在HTTP请求信息流中再次返回到HttpModule中。
IHttpHandler定义了如果要实现一个HTTP请求的处理所必需实现的一些系统约定。HttpHandler与HttpModule不同,一旦定义了自己的HttpHandler类,那么它对系统的HttpHandler的关系将是“覆盖”关系。
当一个HTTP请求经同HttpModule容器传递到HttpHandler容器中时,ASP.NET Framework会调用HttpHandler的ProcessRequest成员方法来对这个HTTP请求进行真正的处理。以一个ASPX页面为例,正是在这里一个ASPX页面才被系统处理解析,并将处理完成的结果继续经由HttpModule传递下去,直至到达客户端。
对于ASPX页面,ASP.NET Framework在默认情况下是交给System.Web.UI.PageHandlerFactory这个HttpHandlerFactory来处理的。所谓一个HttpHandlerFactory,所谓一个HttpHandlerFactory,是指当一个HTTP请求到达这个HttpHandler Factory时,HttpHandlerFactory会提供出一个HttpHandler容器,交由这个HttpHandler容器来处理这个HTTP请求。
一个HTTP请求都是最终交给一个HttpHandler容器中的ProcessRequest方法来处理的。
自定义一个处理请求的类
通过实现IHttpHandler接口可以创建自定义HTTP处理程序,该接口只包含两个方法。通过调用IsReusable,IHttpHandlerFactory可以查询处理程序以确定是否可以使用同一实例为多个请求提供服务。ProcessRequest方法将HttpContext实例用作参数,这使它能够访问Request和Response内部对象。在一个HttpHandler容器中如果需要访问Session,必须实现IRequiresSessionState接口,这只是一个标记接口,没有任何方法。
using System.Web;
using System.Web.SessionState;
namespace MyHttpHandler
{
/// <summary>
/// 自定义HttpHanlder
/// </summary>
public class MyHttpHandler:IHttpHandler,IRequiresSessionState
{
public bool IsReusable
{
get { return true; }
} public void ProcessRequest(HttpContext context)
{
context.Session["user"]="wolfy";
context.Response.Write(context.Session["user"].ToString());
context.Response.Write("httphandler");
}
}
}
在web.config中注册自定义的HttpHanlder
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.5" />
<httpRuntime targetFramework="4.5" /> </system.web>
<system.webServer>
<handlers>
<add name="yhandler" path="*.aspx" verb="*" type="MyHttpHandler.MyHttpHandler,MyHttpHandler"/>
</handlers>
</system.webServer>
</configuration>
name:可以随意
path:指定请求哪些页面使用该自定义的HtppHanlder,支持通配符,如上例中的*.aspx.
type:类的全名称(命名空间+类名),程序集名称。
浏览aspx页面
IHttpHanlder工厂
ASP.NET Framework实际不直接将相关的页面资源HTTP请求定位到一个其内部默认的IHttpHandler容器之上,而定位到了其内部默认的IHttpHandler工厂上。IHttpHandler工厂的作用是对IHttpHandler容器进行调度和管理。
IHttpHandlerFactory接口包含两个方法。GetHandler返回实现IHttpHandler接口的类的实例,ReleaseHandler使工厂可以重用现有的处理程序实例。
namespace MyHttpHandler
{
/// <summary>
/// handler处理工厂
/// </summary>
public class HandlerFactory : IHttpHandlerFactory
{
public IHttpHandler GetHandler(HttpContext context, string requestType, string url, string pathTranslated)
{
PageHandlerFactory factory = (PageHandlerFactory)Activator.CreateInstance(typeof(PageHandlerFactory), true);
IHttpHandler handler = factory.GetHandler(context, requestType, url, pathTranslated);
Page page = handler as Page;
if (page != null)
{
page.Init += new EventHandler(page_Init);
}
return handler;
} private void page_Init(object sender, EventArgs e)
{
HttpContext.Current.Response.Write("IHttpHandlerFactory");
} public void ReleaseHandler(IHttpHandler handler)
{ }
}
}
在web.config中注册工厂
<configuration>
<system.web>
<compilation debug="true" targetFramework="4.5" />
<httpRuntime targetFramework="4.5" /> </system.web>
<system.webServer>
<handlers>
<!--<add name="yhandler" path="*.aspx" verb="*" type="MyHttpHandler.MyHttpHandler,MyHttpHandler"/>-->
<add name="handler1" path="*.aspx" verb="*" type="MyHttpHandler.HandlerFactory,MyHttpHandler"/>
</handlers>
</system.webServer>
</configuration>
浏览
这种方式,可以实现统一验证,在page_init方法中,判断session,判断用户是否登陆了。
总结
通过实现IHttpHandler接口,可以对请求做特殊的处理,比如防盗链的实现,验证码的实现方式等。
参考文章