适用于: Microsoft® ASP.NET 摘要:了解为 ASP.NET Web 页面创建的事件模型,以及 Web 页面转变为 HTML 过程当中的各个阶段。ASP.NET HTTP 运行时负责管理对象管道,这些对象首先将请求的 URL 转换成 Page 类的具体实例,而后再将这些实例转换成纯 HTML 文本。本文将探讨那些做为页面生命周期标志的事件,以及控件和页面编写者如何干预并改变标准行为。(本文包含一些指向英文站点的连接。) 目录 简介 真正的 Page 类 页面的生命周期 执行的各个阶段 小结 简介 对由 Microsoft® Internet 信息服务 (IIS) 处理的 Microsoft® ASP.NET 页面的每一个请求都会被移交到 ASP.NET HTTP 管道。HTTP 管道由一系列托管对象组成,这些托管对象按顺序处理请求,并将 URL 转换为纯 HTML 文本。HTTP 管道的入口是 HttpRuntime 类。ASP.NET 结构为辅助进程中的每一个 AppDomain 建立一个此类的实例。(请注意,辅助进程为每一个当前正在运行的 ASP.NET 应用程序维护一个特定的 AppDomain。) HttpRuntime 类从内部池中获取 HttpApplication 对象,并安排此对象来处理请求。HTTP 应用程序管理器完成的主要任务就是找到将真正处理请求的类。当请求 .aspx 资源时,处理程序就是页面处理程序,即从 Page 继承的类的实例。资源类型和处理程序类型之间的关联关系存储在应用程序的配置文件中。更确切地说,默认的映射集是在 machine.config 文件的 <httpHandlers> 部分定义的。可是,应用程序能够在本地的 web.config 文件中自定义本身的 HTTP 处理程序列表。如下这一行代码就是用来为 .aspx 资源定义 HTTP 处理程序的。 <add verb="*" path="*.aspx" type="System.Web.UI.PageHandlerFactory"/>
扩展名能够与处理程序类相关联,而且更可能是与处理程序工厂类相关联。在全部状况下,负责处理请求的 HttpApplication 对象都会得到一个实现 IHttpHandler 接口的对象。若是根据 HTTP 处理程序来解析关联的资源/类,则返回的类将直接实现接口。若是资源被绑定处处理程序工厂,则还须要额外的步骤。处理程序工厂类实现 IHttpHandlerFactory 接口,此接口的 GetHandler 方法将返回一个基于 IHttpHandler 的对象。 HTTP 运行时是如何结束这个循环并处理页面请求的?ProcessRequest 方法在 IHttpHandler 接口中很是重要。经过对表明被请求页面的对象调用此方法,ASP.NET 结构会启动将生成浏览器输出的进程。 真正的 Page 类 特定页面的 HTTP 处理程序类型取决于 URL。首次调用 URL 时,将构建一个新的类,这个类被动态编译为一个程序集。检查 .aspx 资源的分析进程的结果是类的源代码。该类被定义为命名空间 ASP 的组成部分,而且被赋予了一个模拟原始 URL 的名称。例如,若是 URL 的终点是 page.aspx,则类的名称就是 ASP.Page_aspx。不过,类的名称能够经过编程方式来控制,方法是在 @Page 指令中设置 ClassName 属性。 HTTP 处理程序的基类是 Page。这个类定义了由全部页面处理程序共享的方法和属性的最小集合。Page 类实现 IHttpHandler 接口。 在不少状况下,实际处理程序的基类并非 Page,而是其余的类。例如,若是使用了代码分离,就会出现这种状况。代码分离是一项开发技术,它能够将页面所需的代码隔离到单独的 C# 和 Microsoft Visual Basic® .NET 类中。页面的代码是一组事件处理程序和辅助方法,这些处理程序和方法真正决定了页面的行为。可使用 <script runat=server> 标记对此代码进行内联定义,或者将其放置在外部类(代码分离类)中。代码分离类是从 Page 继承并使用额外的方法的类,被指定用做 HTTP 处理程序的基类。 还有一种状况,HTTP 处理程序也不是基于 Page 的,即在应用程序配置文件的 <pages> 部分中,包含了 PageBaseType 属性的从新定义。 <pages PageBaseType="Classes.MyPage, mypage" />
PageBaseType 属性指明包含页面处理程序的基类的类型和程序集。从 Page 导出的这个类能够自动赋予处理程序扩展的自定义方法和属性集。 页面的生命周期 彻底识别 HTTP 页面处理程序类后,ASP.NET 运行时将调用处理程序的 ProcessRequest 方法来处理请求。一般状况下,无需更改此方法的实现,由于它是由 Page 类提供的。 此实现将从调用为页面构建控件树的 FrameworkInitialize 方法开始。FrameworkInitialize 方法是 TemplateControl 类(Page 自己今后类导出)的一个受保护的虚拟成员。全部为 .aspx 资源动态生成的处理程序都将覆盖 FrameworkInitialize。在此方法中,构建了页面的整个控件树。 接下来,ProcessRequest 使页面经历了各个阶段:初始化、加载视图状态信息和回发数据、加载页面的用户代码以及执行回发服务器端事件。以后,页面进入显示模式:收集更新的视图状态,生成 HTML 代码并随后将代码发送到输出控制台。最后,卸载页面,并认为请求处理完毕。 在各个阶段中,页面会触发少数几个事件,这些事件能够由 Web 控件和用户定义的代码截取并进行处理。其中的一些事件是嵌入式控件专用的,所以没法在 .aspx 代码级进行处理。 要处理特定事件的页面应该明确注册一个适合的处理程序。不过,为了向后兼容早期的 Visual Basic 编程风格,ASP.NET 也支持隐式事件挂钩的形式。默认状况下,页面会尝试将特定的方法名称与事件相匹配,若是实现匹配,则认为此方法就是匹配事件的处理程序。ASP.NET 提供了六种方法名称的特定识别,它们是 Page_Init、Page_Load、Page_DataBind、Page_PreRender 和 Page_Unload。这些方法被认为是由 Page 类提供的相应事件的处理程序。HTTP 运行时会自动将这些方法绑定到页面事件,这样,开发人员就没必要再编写所需的粘接代码了。例如,若是命名为 Page_Load 的方法绑定到页面的 Load 事件,则可省去如下代码。 this.Load += new EventHandler(this.Page_Load);
对特定名称的自动识别是由 @Page 指令的 AutoEventWireup 属性控制的。若是该属性设置为 false,则要处理事件的全部应用程序都须要明确链接到页面事件。不使用自动绑定事件的页面性能会稍好一些,由于不须要额外匹配名称与事件。请注意,全部 Microsoft Visual Studio® .NET 项目都是在禁用 AutoEventWireup 属性的状况下建立的。可是,该属性的默认设置是 true,即 Page_Load等方法会被识别,并被绑定到相关联的事件。 下表中按顺序列出了页面的执行包括的几个阶段,执行的标志是一些应用程序级的事件和/或受保护并可覆盖的方法。 表 1:ASP.NET 页面生命中的关键事件
阶段 |
页面事件 |
可覆盖的方法 |
页面初始化 |
Init |
|
加载视图状态 |
|
LoadViewState |
处理回发数据 |
|
任意实现 IPostBackDataHandler 接口的控件中的 LoadPostData 方法 |
加载页面 |
Load |
|
回发更改通知 |
|
任意实现 IPostBackDataHandler 接口的控件中的 RaisePostDataChangedEvent 方法 |
处理回发事件 |
由控件定义的任意回发事件 |
任意实现 IPostBackDataHandler 接口的控件中的 RaisePostBackEvent 方法 |
页面显示前阶段 |
PreRender |
|
保存视图状态 |
|
SaveViewState |
显示页面 |
|
Render |
卸载页面 |
Unload |
|
以上所列的阶段中有些在页面级是不可见的,而且仅对服务器控件的编写者和要建立从 Page 导出的类的开发人员有意义。Init、Load、PreRender、Unload,再加上由嵌入式控件定义的全部回发事件,就构成了向外发送页面的各个阶段标记。 执行的各个阶段 页面生命周期中的第一个阶段是初始化。这个阶段的标志是 Init 事件。在成功建立页面的控件树后,将对应用程序触发此事件。换句话说,当 Init 事件发生时,.aspx 源文件中静态声明的全部控件都已实例化并采用各自的默认值。控件能够截取 Init 事件以初始化在传入的 Web 请求的生命周期内所需的全部设置。例如,这时控件能够加载外部模板文件或设置事件的处理程序。请注意,这时视图状态信息尚不可用。 初始化以后,页面框架将加载页面的视图状态。视图状态是名称/值对的集合,在此集合中,控件和页面自己存储了对全部 Web 请求都必须始终有效的所有信息。视图状态表明了页面的调用上下文。一般,它包含上次在服务器上处理页面时控件的状态。首次在会话中请求页面时,视图状态为空。默认状况下,视图状态存储在静默添加到页面的隐藏字段中,该字段的名称是 __VIEWSTATE。经过覆盖 LoadViewState 方法(Control 类的受保护、可覆盖方法),组件开发人员能够控制视图状态的存储方式以及视图状态的内容映射到内部状态的方式。 有些方法(如 LoadPageStateFromPersistenceMedium 以及其对应的 SavePageStateToPersistenceMedium),能够用来将视图状态加载并保存到其余存储介质(例如会话、数据库或服务器端文件)中。与 LoadViewState 不一样,上述方法只能在从 Page 导出的类中使用。 存储视图状态以后,页面树中控件的状态与页面最后一次显示在浏览器中的状态相同。下一步是更新它们的状态以加入客户端的更改。处理回发数据阶段使控件有机会更新其状态,从而准确反映客户端相应的 HTML 元素的状态。例如,服务器的 TextBox 控件对应的 HTML 元素是 <input type=text>。在回发数据阶段,TextBox 控件将检索 <input> 标记的当前值,并使用该值来刷新本身内部的状态。每一个控件都要从回发的数据中提取值并更新本身的部分属性。TextBox 控件将更新它的 Text 属性,而 CheckBox 控件将刷新它的 Checked 属性。服务器控件和 HTML 元素的对应关系能够经过两者的 ID 找到。 在处理回发数据阶段的最后,页面中的全部控件的状态都将使用客户端输入的更改来更新前一状态。这时,将对页面触发 Load 事件。 页面中可能会有一些控件,当其某个敏感属性在两个不一样的请求中被修改时,须要完成特定的任务。例如,若是 TextBox 控件的文本在客户端被修改,则此控件将触发 TextChanged 事件。每一个控件在其一个或多个属性被修改成客户端输入的值时均可以决定触发相应的事件。对于这些更改对其很是关键的控件,控件实现 IPostBackDataHandler 接口,此接口的 LoadPostData 方法是在 Load 事件后当即调用的。经过对 LoadPostData 方法进行编码,控件将验证自上次请求后是否发生了关键更改,并触发本身的更改事件。 页面生命周期中的关键事件是被调用以执行服务器端代码的事件,此代码与客户端触发的事件相关联。当用户单击按钮时,将回发页面。回发值的集合中包括启动整个操做的按钮的 ID。若是控件实现 IPostBackEventHandler 接口(如按钮和连接按钮),页面框架将调用 RaisePostBackEvent 方法。此方法的行为取决于控件的类型。就按钮和连接按钮而言,此方法将查找 Click 事件处理程序并运行相关的委托。 处理完回发事件以后,页面就能够显示了。这个阶段的标志是 PreRender 事件。控件能够利用这段时间来执行那些须要在保存视图状态和显示输出的前一刻执行的更新操做。下一个状态是 SaveViewState,在此状态中,全部控件和页面自己都将更新本身 ViewState 集合的内容。而后,将获得序列化、散列、Base64 编码的视图状态,并且此视图状态与隐藏字段 __VIEWSTATE 相关联。 经过覆盖 Render 方法能够改变各个控件的显示机制。此方法接受 HTML 书写器对象,并使用此对象来积累全部要为控件生成的 HTML 文本。Page 类的 Render 方法的默认实现包括对全部成员控件的递归调用。对于每一个控件,页面都将调用 Render 方法,并缓存 HTML 输出。 页面生命中的最后一个标志是 Unload 事件,在页面对象消除以前发生。在此事件中,您应该释放全部可能占用的关键资源(例如文件、图形对象、数据库链接等)。 在此事件以后,也就是最后,浏览器接收 HTTP 响应数据包并显示页面。 小结 ASP.NET 页面对象模型因其事件机制而显得格外新颖独特。Web 页面由控件组成,这些控件既能够产生丰富的基于 HTML 的用户界面,又能够经过事件与用户交互。之前,在 Web 应用程序的上下文中设置事件模型是件有挑战性的工做。可咱们惊奇的看到,客户端生成的事件能够由服务器端的代码来解决,并且只进行一些相应的修改后,此过程仍能够输出相同的 HTML 页面。 掌握这个模型对于了解页面生命周期的各个阶段,以及页面对象如何被 HTTP 运行时实例化并使用是很是重要的。 关于做者 Dino Esposito 是一位来自意大利罗马的培训教师和顾问。做为 Wintellect 团队的成员,Dino 专门研究 ASP.NET 和 ADO.NET,主要在欧洲和美国从事教学和咨询工做。此外,Dino 还负责管理 Wintellect 的 ADO.NET 课件,并为 MSDN 期刊的“Cutting Edge”专栏撰写文章。要与他联系,请向 dinoe@wintellect.com 发送电子邮件。 |