了解 IHttpHandler
1 、概述
说明:HttpHandler是一个HTTP请求的真正处理中心。在HttpHandler容器中,ASP.NET Framework才调用HttpHandler的ProcessRequest成员方法来对这个HTTP请求进行真正的处理,真正地对客户端请求的服务器页面做出编译和执行,并将处理过后的信息附加在HTTP请求信息流中再次返回到HttpModule中。
2、举例
以一个aspx页面为例,正是在HttpHandler这里,一个aspx页面才被系统处理解析,并将处理完成的结果继续经由HttpModule传递下去,直至到达客户端。当然,对于aspx页面,ASP.NET Framework在默认情况下是交给System.Web.UI.PageHandlerFactory这个HttpHandlerFactory来处理的。当一个HTTP请求到达这个HttpHandlerFactory时,HttpHandlerFactory会提供出一个HttpHandler容器,交由这个HttpHandler容器来处理这个HTTP请求。 一个HTTP请求都是最终交给一个HttpHandler容器中的ProcessRequest方法来处理的。
3、HttpHandler
(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,而不再使用原先默认的或者指定的.
4、HttpHandlerFactory
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工厂处理好了。
5、HttpHandler和HttpModule的区别
主要有两点:
(1)先后次序.先IHttpModule,后IHttpHandler,IHttpHandler处理结束后再交给IHttpModule;
(2)对请求的处理上:
IHttpModule是属于大小通吃类型,无论客户端请求的是什么文件,都会调用到它;例如aspx,html,rar的请求;
IHttpHandler则属于挑食类型,只有asp.net注册过的文件类型(例如aspx,ascx,asmx等等)才会轮到调用它。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义