用winform的方式操纵webform--浅谈IHttpHandler |
|
好久以前看过一篇有关自定义IHttpHandler的文章,当时就想写点自己的体会,一直没空。今天就说说这个吧。首先谈谈asp.net的一些参数传递和页面定向的方式 第一,asp.net是用Page.Navigate()调用新页面的URL。Page.Navigate()向浏览器返回了一个http状态码302,使得浏览器向server请求那个新的URL。这种导航方法导致每次客户请求都需两次在client和server之间往返。第二,任何要传递到新页面的信息都需作为URL的参数或存储在Session中或存储在数据库中以便新页面得到这些信息。传统的asp开发人员很习惯这种做法,但其他的web编程人员则有一些更高级的方法。但是很明显的这两个页面是有依赖性的,而依赖性是编译器捕捉不到的也是不容易在设计阶段建模的。所以在debug时,参数是否被正确的传递就只有我们自己检查了。再有传统的数据传递方式有可能会暴露一些关键的数据。更为关键的是这使得面向对象的设计变得很复杂。当然我并没有否定传统的方式,我这是在强调asp.net缺乏对服务器端的多页面之间的通信的支持。 但是我们可以自定义httphandler来扩充这种支持。 开始之前,我们先来看一下asp.net怎样处理页面请求,asp.net是通过System.Web.UI.PageHandlerFactory类的实例来处理。PageHandlerFactory依赖于另一个类PageParser。一个页面在第一次请求时PageParser把它编译成一个真正的.Net的类并cache已编译的实例 HttpApplication | |1.GetHandler() | PageHandlerFactory | |2.GetCompiledPageInstance() | PageParser 我们要做的第一步是override由asp.net提供的页面生成过程。所以我们必须创建一个类实现IHttpHandlerFactory接口,我们需要实现的只有两个方法,一个是GetHandler(),它是在一个HTTP Request开始时被调用;另一个是ReleaseHandler(),调用可以让IHttpHandlerfactory进行清理工作。 例子: using System; using System.Web; using System.Web.UI; namespace Michael.Web.HttpHandler { /// <summary> /// 一个简单的HttpHandler /// </summary> public class Factory:System.Web.IHttpHandlerFactory { public IHttpHandler GetHandler(HttpContext context, string url, string path, string objectName) { return Controller.getViewObject(path, objectName, context); } public void ReleaseHandler(IHttpHandler Handler) { } } } 注意:GetHandler()必须返回一个实现IHttpHandler接口的对象,Controller是另一个我们自定义的类 接下来我们要修改web.config <configuration> <httphandlers> <add verb="*" path="*.aspx" type="Michael.Web.HttpHandler.Factory,Michael.Web.HttpHandler" /> </httphandlers> </configuration> 我们这里还是采用.aspx作为后缀名,其实你可以自定义这个后缀名 例子:Controller.cs namespace Michael.Web.HttpHandler { using System; using System.Web; using System.Web.UI; using System.Collections.Specialized; /// <summary> /// Controller继承了System.Web.UI.Page作为ASP.NET页面的handler /// </summary> public class Controller : System.Web.UI.Page { public Controller() { this.Init += new System.EventHandler( Base_Init ); } protected void Page_Load( object sender, EventArgs e ) { //将提交页面的对象名存储在hidden里 string a_ObjectName = this.GetType().Name; a_ObjectName = a_ObjectName.Substring ( 0, a_ObjectName.Length - 5 ); this.RegisterHiddenField( LAST_VIEW_OBJECT, a_ObjectName ); } protected void Page_Init( object sender, EventArgs e ) { this.Load += new System.EventHandler ( this.Base_Load ); } ////////////////// ///创建页面实例,允许服务器端通讯 ////////////////// protected IHttpHandler createInstanceOf( string ObjectName ) { string a_Path = Request.ApplicationPath + "/" + ObjectName; return Controller.getPage( a_Path, this.Context.Request.MapPath(ObjectName) , Context ); } protected virtual IHttpHandler getNextPage() { return null; } protected override void Render( HtmlTextWriter writer ) { IHttpHandler aHandler = this.getNextPage(); if( aHandler == null ) { base.Render ( writer ); } else { if( aHandler is Controller ) { ( ( Controller )aHandler ).overrideLoadState = true; } aHandler.ProcessRequest ( this.Context ); } } ///////////////// ///先检查hidden看是否有提交页面名称,如没有则返回浏览器请求的URL ///////////////// public static IHttpHandler getViewObject( string Path, string ObjectName , HttpContext Context ) { string a_CurrentView = Context.Request.Form[LAST_VIEW_OBJECT]; string a_Path; if( a_CurrentView == null || a_CurrentView.Length == 0 ) { a_CurrentView = ObjectName; a_Path = Path; } else { a_CurrentView += ".aspx"; a_Path = Context.Request.ApplicationPath + "/" + a_CurrentView; a_CurrentView = Context.Server.MapPath( a_CurrentView ); } return Controller.getPage( a_Path, a_CurrentView, Context ); } private static IHttpHandler getPage( string Path, string ObjectName , HttpContext Context ) { IHttpHandler a_Page = null; try { a_Page = PageParser.GetCompiledPageInstance( Path , ObjectName , Context ); } catch(Exception e) { // 返回错误页面 } return a_Page; } protected override object LoadPageStateFromPersistenceMedium() { if( overrideLoadState ) { ViewState.Clear(); return null; } return base.LoadPageStateFromPersistenceMedium(); } protected override NameValueCollection DeterminePostBackMode() { if(overrideLoadState) { return null; } else { return base.DeterminePostBackMode(); } } protected bool overrideLoadState = false; protected const string LAST_VIEW_OBJECT = "ControllerLASTVIEWOBJECT"; } } 这样我们就建立了一个自定义的httphandler,在你的webform中继承Controller并重载getNextPage方法 看一看我们可以怎样操纵页面呢 protected override IHttpHandler getNextPage() { if(this.IsValid && this.IsPostBack) { Second SecondPage = this.createInstanceOf("second.aspx") as Second;//是不是很象winform呀 Infomation FirstPage = new Infomation(); FirstPage.EmailAddress = txtEmail.Text; FirstPage.FirstName = txtFirstName.Text; FirstPage.LastName = txtLastName.Text; SecondPage.Infomation = FirstPage; return SecondPage; } else { return null; } } |
正文
用winform的方式操纵webform--浅谈IHttpHandler2006-08-10 11:44:00
【评论】 【打印】 【字体:大 中 小】 本文链接:http://blog.pfan.cn/Csharpsky/17443.html
阅读(2538) | 评论(0)
版权声明:编程爱好者网站为此博客服务提供商,如本文牵涉到版权问题,编程爱好者网站不承担相关责任,如有版权问题请直接与本文作者联系解决。谢谢!
评论