HttpHandler初探

一、HttpHandler简介

HttpHandler是asp.net真正处理Http请求的地方。在这个HttpHandler容器中,ASP.NET Framework才真正地对客户端请求的服务器页面做出编译和执行,并将处理过后的信息附加在HTTP请求信息流中再次返回到HttpModule中。

IHttpHandler在MSDN中的定义:使用自定义的HTTP处理程序同步处理HTTP Web请求而实现的协定。(注意:这里写的很清楚,是同步HTTP请求。如果是异步的话就要使用IHttpAsyncHandler接口)。他包含一个属性IsReusable,用于获取当前IHttpHandler实例是否可用,一般设置为true。一个方法ProcessRequest(HttpContext context),进行实际的操作过程。

HttpHandler与HttpModule不同,一旦定义了自己的HttpHandler类,那么它对系统的HttpHandler的关系将是“覆盖”关系。

二、IHttpHandler的处理过程

当一个HTTP请求经过HttpModule容器传递到HttpHandler容器中时,ASP.NET Framework会调用HttpHandler的ProcessRequest成员方法来对这个HTTP请求进行真正的处理。以一个ASPX页面为例,正是在这里一个ASPX页面才被系统处理解析,并将处理完成的结果继续经由HttpModule传递下去,直至到达客户端。

对于ASPX页面,ASP.NET Framework在默认情况下是交给System.Web.UI.PageHandlerFactory这个HttpHandlerFactory来处理的。所谓一个HttpHandlerFactory,是指当一个HTTP请求到达这个HttpHandlerFactory时,HttpHandlerFactory会提供出一个HttpHandler容器,交由这个HttpHandler容器来处理这个HTTP请求。

一个HTTP请求都是最终交给一个HttpHandler容器中的ProcessRequest方法来处理的。

三、一个简单的自定义IHttpHandler

实现一个自定义的IHttpHandler需要两步:

第一:定义一个实现了IHttpHandler的类,并且实现其ProcessRequest方法。在一个HttpHandler容器中如果需要访问Session,必须实现IRequiresSessionState接口,这只是一个标记接口,没有任何方法。


using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Web;
namespace MyHttpHandler {     public class HelloHttpHandler:IHttpHandler     {         public bool IsReusable          {             get { return true; }         }
        public void ProcessRequest(HttpContext context)          {             context.Response.Write("我是经过HelloHttpHandler处理的!");         }     } }

第二:在web.config中注册这个类,并且指定Handler处理的请求类型

把此节点插入system.web节点中

  <httpHandlers>       <!--path中指定的是执行type中HttpHandler的访问路径。此路径可以带后缀也可以不带后缀。如果path配置为*,则会对所有的请求执行此HttpHandler-->       <add  verb="*" path="httphandlerlearn.html" type="MyHttpHandler.HelloHttpHandler,MyHttpHandler"/>     </httpHandlers>

这时,访问httphandlerlearn.html就会显示“我是经过HelloHttpHandler处理的!”,如下:

五、HttpHandlerFactory

ASP.NET Framework实际不直接将相关页面的HTTP请求定位到一个其内部默认的IHttpHandler容器之上,而定位到了其内部默认的IHttpHandlerFactory上。IHttpHandlerFactory的作用是对IHttpHandler容器进行调度和管理。

如果我们自定义的HttpHandler比较多,会在Web.config中注册很多HttpHandler类,这时把这些HttpHandler通过一个HttpHandlerFactory来集中管理就显的非常必要,此时只需要在Web.config中注册此HttphandlerFactory即可。

IHttpHandlerFactory接口包含两个方法。GetHandler返回实现IHttpHandler接口的类的实例,ReleaseHandler使工厂可以重用现有的处理程序实例。

实现一个自定义的HttphandlerFactory同样需要两步:

第一:定义一个实现了IHttpHandlerFactory接口的类,并实现其GetHandler方法。在GetHandler中,我们可以根据具体业务选择不同的工厂模式实现方式,下面的代码中包括了通过判断后缀实现的简单工厂模式和通过反射实现的工厂模式两种实现方式。


using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Web; using System.IO;
namespace MyHttpHandler {     public class MyHttpHandlerFactory:IHttpHandlerFactory     {         public IHttpHandler GetHandler(HttpContext context, string requestType, string url, string pathTranslated)         {             #region 通过判断后缀实现简单工厂模式             ////获取文件服务器物理路径             //string path = context.Request.PhysicalPath;             ////获取文件名后缀名             //string exstention = Path.GetExtension(path).ToLower();
            //if (exstention == ".html")             //{             //    return new HtmlHttpHandler();             //}             //else if (exstention == ".xml")             //{             //    return new XMLHttpHandler();             //}             //else             //{             //    return null;             //}             #endregion
            #region 通过反射实现工厂模式             string handlerName = url.Substring(url.LastIndexOf("/") + 1);             string className = handlerName.Substring(0, handlerName.IndexOf("."));             string fullClassName = "MyHttpHandler." + className;
            object h = null;
            // 采用动态反射机制创建相应的IHttpHandler实现类。             h = Activator.CreateInstance(Type.GetType(fullClassName));
            return (IHttpHandler)h;
            #endregion
        }
        public void ReleaseHandler(IHttpHandler handler)          {                      }     } }

第二:在web.config中注册此类

    <httpHandlers>     <add  verb="*" path="HtmlHttpHandler.html,XMLHttpHandler.xml" type="MyHttpHandler.MyHttpHandlerFactory,MyHttpHandler"/>     </httpHandlers>

此时,访问HtmlHttpHandler.html和XMLHttpHandler.xml的效果如下:

四、HttpHandler的实际应用

因为HttpHandler是Asp.net真正处理请求的地方,所以我们可以在HttpHandler中对请求做一些自定义的处理,甚至彻底改变请求的预期输出。下面列举了几个HttpHandler的应用:

一、图片防盗链应用。

有时我们自己web站点中的图片不想被其他站点引用,或者不想让人知道图片地址后直接通过浏览器访问,这就是图片的防盗链。下面是通过HttpHandler实现的图片防盗链:


using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Web;
namespace MyHttpHandler {     /// <summary>     /// 图片防盗链HttpHandler     /// </summary>     public class PictureHttpHandler : IHttpHandler     {         public bool IsReusable         {             get { return true; }         }
        public void ProcessRequest(HttpContext context)         {             //站点的域名             string myDomain = "localhost";
            if (context.Request.UrlReferrer == null ||                 context.Request.UrlReferrer.Host.ToLower().IndexOf(myDomain) < 0)             {                 //如果是通过浏览器直接访问或者是通过其他站点访问过来的,则显示“资源不存在”图片                 context.Response.ContentType = "image/JPEG";                 context.Response.WriteFile(context.Request.PhysicalApplicationPath+"/images/noimg.jpg");             }             else {                 //如果是通过站内访问的,这正常显示图片                 context.Response.ContentType = "image/JPEG";                 context.Response.WriteFile(context.Request.PhysicalPath);             }                     }     } }

把下面的项加到web.config中的httphandler节点中:

<add  verb="*" path="*.jpg" type="MyHttpHandler.PictureHttpHandler,MyHttpHandler"/>

正常访问default页面时:

通过图片地址直接访问时:

二、生成验证码


using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Drawing; using System.Drawing.Imaging; using System.Web;
namespace MyHttpHandler {     /// <summary>     /// 验证码Handler     /// </summary>     public class ValidateCodeHttpHandler : IHttpHandler     {         public void ProcessRequest(HttpContext context)         {             context.Response.ContentType = "image/gif";             //建立Bitmap对象,绘图                     Bitmap basemap = new Bitmap(200, 60);             Graphics graph = Graphics.FromImage(basemap);             graph.FillRectangle(new SolidBrush(Color.White), 0, 0, 200, 60);             Font font = new Font(FontFamily.GenericSerif, 48, FontStyle.Bold, GraphicsUnit.Pixel);             Random r = new Random();             string letters = "ABCDEFGHIJKLMNPQRSTUVWXYZ";             string letter;             StringBuilder s = new StringBuilder();             //添加随机的五个字母                     for (int x = 0; x < 5; x++)             {                 letter = letters.Substring(r.Next(0, letters.Length - 1), 1);                 s.Append(letter);                 graph.DrawString(letter, font, new SolidBrush(Color.Black), x * 38, r.Next(0, 15));             }             //混淆背景                     Pen linePen = new Pen(new SolidBrush(Color.Black), 2);             for (int x = 0; x < 6; x++)                 graph.DrawLine(linePen, new Point(r.Next(0, 199), r.Next(0, 59)),                     new Point(r.Next(0, 199), r.Next(0, 59)));             //将图片保存到输出流中                           basemap.Save(context.Response.OutputStream, ImageFormat.Gif);             //context.Session["CheckCode"] = s.ToString();                //如果没有实现IRequiresSessionState,则这里会出错,也无法生成图片                      context.Response.End();         }         public bool IsReusable          {                get { return true; }          }     } }

把下面的项加到web.config中的httphandler节点中:

<add  verb="*" path="validatevode" type="MyHttpHandler.ValidateCodeHttpHandler,MyHttpHandler"/>

访问validatevode时:

源代码:/Files/yourshj/HttpHandler.rar

HttpHandler的更多相关文章

  1. ASP.NET Core的路由[1]:注册URL模式与HttpHandler的映射关系

    ASP.NET Core的路由是通过一个类型为RouterMiddleware的中间件来实现的.如果我们将最终处理HTTP请求的组件称为HttpHandler,那么RouterMiddleware中间 ...

  2. [转]ASP.NET应用程序生命周期趣谈(四) HttpHandler和页面生命周期

    在之前的三篇文章中,我们还算简明扼要的学习了asp.net的整个生命周期,我们知道了一个Request进来以后先去ISAPI Filter,发现是asp.net程序后又ASPNET_ISAPI.dll ...

  3. httphandler和httpmodule的区别

    ASP.Net处理Http Request时,使用Pipeline(管道)方式,由各个HttpModule对请求进行处理,然后到达 HttpHandler,HttpHandler处理完之后,仍经过Pi ...

  4. 你不知道的HttpHandler相关知识

    一.关于IHttpHandler.IsReusable 很多人不明白,这哥们到底干嘛的,估计是微软最初的一个想法--让一个对象可以一直不断地被重复使用 ,但这个想法不成熟,会带来很多隐藏问题,一个对象 ...

  5. HttpModule & HttpHandler

    ASP.NET 处理请求的过程 inetinfo.exe:www 服务进程,IIS 服务 和 ASPNET_ISAPI.dll 都寄存在此进程中. ASPNET_ISAPI.dll:处理 .aspx ...

  6. 选择HttpHandler还是HttpModule?

    阅读目录 开始 理解ASP.NET管线 理解HttpApplication 理解HttpHandler 理解HttpModule 三大对象的总结 案例演示 如何选择? 最近收到几个疑问:HttpHan ...

  7. ASP.NET-自定义HttpModule与HttpHandler

    在之前的ASP.NET是如何在IIS下工作的这篇文章中介绍了ASP.NET与IIS配合工作的机制,在http请求经过一系列处理后,最后到达ASP.NET管道中,这时,就是Http Modules和Ht ...

  8. 我心中的核心组件~HttpHandler和HttpModule实现图像的缩放与Url的重写

    回到目录 说在前 对于资源列表页来说,我们经常会把图像做成N多种,大图,小图,中图等等,很是麻烦,在数据迁移时,更是一种痛快,而如果你把图像资源部署到nginx上,那么这种图像缩放就变得很容易了,因为 ...

  9. 关于HttpHandler的相关知识总结

    一.关于IHttpHandler.IsReusable 很多人不明白,这哥们到底干嘛的,估计是微软最初的一个想法--让一个对象可以一直不断地被重复使用 ,但这个想法不成熟,会带来很多隐藏问题,一个对象 ...

随机推荐

  1. 为现有图像处理程序添加读写exif的功能

    为现有图像处理程序添加读取exif的功能 exif是图片的重要参数,在使用过程中很关键的一点是exif的数据能够和图片一起存在.exif的相关功能在操作系统中就集成了,在csharp中也似乎有了实现. ...

  2. Apache开启状态查看页面(原创贴-转载请注明出处)

    =================写在前面的话================== 场景描述:有时候我们需要查看apache的运行状态,只需要开启apache的status功能就可以实现,但是stat ...

  3. 一看便知spring+quartz定时任务

    这是我经过网上收集然后加上自己的测试写的,以便大家使用 标配:已测 注意需要的包:(在已经配置spring 的情况下) quartz-all-1.6.jar        spring-context ...

  4. google浏览器打开报出文件不可读解决方案

    1.打开*.desktop文件 gedit ~/.local/share/applications/name.desktop 在文件中做改动: The .desktop file should loo ...

  5. bean找不到异常

    和这种的 原因: 这些都是因为bean注入的时候没有找个要注入的bean 解决办法: 1.查看dubbo文件中,暴露接口是否引入bean 2.如果有引入,查看引入路径和类是否存在.

  6. File API 读取上传的文件

    1, 在html 文档中,<input type="file"> 我们可以选择文件进行上传,但这时只能上传一个文件.如果加上multiple 属性,可以上传多个文件,上 ...

  7. MFC编程入门之二十一(常用控件:编辑框Edit Control)

    上一节讲了静态文本框,本节讲的是编辑框(Edit Control)同样是一种很常用的控件,我们可以在编辑框中输入并编辑文本.在前面加法计算器的例子中已经演示了编辑框的基本应用.下面具体讲解编辑框的使用 ...

  8. Intellij 打开就闪退或关闭

    找到安装目录的bin目录,搜索vmoptions可以看到两个文件, idea.exe.vmoptions    idea64.exe.vmoptions 这两个文件就是IDEA的一些配置文件,带64位 ...

  9. C# for循环 循环中先求长度还是使用list.Count,哪个效率高

    这种问法可能问的不好.应该说成是第一种方法完全没有必要,很多人可能以为那样会为代码带来效率,而实际上是不会给效率带来任何提升. 因为事实上,索引器内部,为了安全期间,还是会去求整个list的count ...

  10. Linux基础知识

    1.url中不写端口号,默认就是80端口:本机是127.0.0.1或者localhost 2.用户管理 查看当前用户: id:可以查看当前用户:whoami:查看当前的用户:who:可以查看当前已经登 ...