ASP.NET應(yīng)用程序與頁面生命周期
在本文中,我們將了解不同的事件,ASP.NET 應(yīng)用程序的生命周期以瀏覽器向 Web 服務(wù)器(對于 ASP.NET 應(yīng)用程序,通常為 IIS)發(fā)送請求為起點(diǎn),直至將請求結(jié)果返回至瀏覽器結(jié)束。在這個(gè)過程中,首先我們需要了解ASP.NET請求的2個(gè)大致的步驟。其次我們將詳細(xì)了解 'httphandler ',' httpmodule和 asp.net 頁面對象(Page)中不同的事件的執(zhí)行順序,邏輯。
二個(gè)步驟的過程:
asp.net請求處理,2步的過程如下所示,用戶發(fā)送一個(gè)請求到IIS 服務(wù)器:
1、asp.net創(chuàng)建一個(gè)運(yùn)行時(shí),可以處理請求。換句話說,它創(chuàng)建應(yīng)用程序?qū)ο螅埱?,響?yīng)和上下文對象處理請求。
2、運(yùn)行時(shí)一旦被創(chuàng)建,請求處理,通過一系列的事件處理模塊,Handler處理和頁面對象。簡稱MHPM (Module, handler, page and Module event)。
ASP.NET 應(yīng)用程序生命周期的各個(gè)階段:
步驟1:用戶從 Web 服務(wù)器請求應(yīng)用程序資源 ASP.NET 應(yīng)用程序的生命周期以瀏覽器向 Web 服務(wù)器(對于 ASP.NET 應(yīng)用程序,通常為 IIS)發(fā)送請求為起點(diǎn)。ASP.NET 是 Web 服務(wù)器下的 ISAPI 擴(kuò)展。Web 服務(wù)器接收到請求時(shí),會對所請求的文件的文件擴(kuò)展名進(jìn)行檢查,確定應(yīng)由哪個(gè) ISAPI 擴(kuò)展處理該請求,然后將該請求傳遞給合適的 ISAPI 擴(kuò)展。ASP.NET 處理已映射到其上的文件擴(kuò)展名,如 .aspx、.ascx、.ashx 和 .asmx。
步驟2:ASP.NET 接收對應(yīng)用程序的***個(gè)請求 當(dāng) ASP.NET 接收到對應(yīng)用程序中任何資源的***個(gè)請求時(shí),名為 ApplicationManager 的類會創(chuàng)建一個(gè)應(yīng)用程序域。應(yīng)用程序域?yàn)槿肿兞刻峁?yīng)用程序隔離,并允許單獨(dú)卸載每個(gè)應(yīng)用程序。在應(yīng)用程序域中,將為名為 HostingEnvironment 的類創(chuàng)建一個(gè)實(shí)例,該實(shí)例提供對有關(guān)應(yīng)用程序的信息(如存儲該應(yīng)用程序的文件夾的名稱)的訪問。
步驟3:為每個(gè)請求創(chuàng)建 ASP.NET 核心對象 創(chuàng)建了應(yīng)用程序域并對 HostingEnvironment 對象進(jìn)行了實(shí)例化之后,ASP.NET 將創(chuàng)建并初始化核心對象,如 HttpContext、HttpRequest 和 HttpResponse。HttpContext 類包含特定于當(dāng)前應(yīng)用程序請求的對象,如 HttpRequest 和 HttpResponse 對象。HttpRequest 對象包含有關(guān)當(dāng)前請求的信息,包括 Cookie 和瀏覽器信息。HttpResponse 對象包含發(fā)送到客戶端的響應(yīng),包括所有呈現(xiàn)的輸出和 Cookie。
步驟4:將 HttpApplication 對象分配給請求初始化所有核心應(yīng)用程序?qū)ο笾?,將通過創(chuàng)建 HttpApplication 類的實(shí)例啟動(dòng)應(yīng)用程序。如果應(yīng)用程序具有 Global.asax 文件,則 ASP.NET 會創(chuàng)建 Global.asax 類(從 HttpApplication 類派生)的一個(gè)實(shí)例,并使用該派生類表示應(yīng)用程序。
注:***次在應(yīng)用程序中請求 ASP.NET 頁或進(jìn)程時(shí),將創(chuàng)建 HttpApplication 的一個(gè)新實(shí)例。不過,為了盡可能提高性能,可對多個(gè)請求重復(fù)使用 HttpApplication 實(shí)例。
步驟5: 由 HttpApplication 管線處理請求 MHPM (Module, handler, page and Module event)。
#p#
使用MHPM (Module, handler, page and Module event) 事件處理請求
一旦 HttpApplication對象創(chuàng)建,在處理該請求時(shí)將由 HttpApplication 類執(zhí)行。我們來了解下HttpModule HttpHandlers。
1、如果你想通過*.aspx,*.html文件來處理程序邏輯,那么你需要使用HttpHandler,換句話說,httphandler是一個(gè)擴(kuò)展的處理器。
2、如果你想使用ASP.NET管道來處理程序邏輯,你需要使用HttpModule 換句話說,httpmodule是一個(gè)事件處理器。
如何處理請求。mhpm有4個(gè)重要的步驟:
步驟1(M: HttpModule):客戶端開始請求處理。在asp.net引擎和httpmodule中的事件可以用來處理用戶自己定義的邏輯。有6個(gè)重要的事件,你可以在你的頁面對象被創(chuàng)建之前調(diào)用 Begin Request,authenticaterequest,authorizerequest,resolverequestcache,acquirerequeststate與prerequesthandlerexecute。
步驟2(H: ‘HttpHandler’):一旦上述6個(gè)事件被執(zhí)行完,asp.net引擎調(diào)用processrequest事件,如果已實(shí)現(xiàn)httphandler在您的項(xiàng)目。
步驟3(P: ASP.NET page):一旦httphandler執(zhí)行,asp.net頁面對象被創(chuàng)建。而asp.net頁面對象被創(chuàng)建,Page對象的事件將被調(diào)用,這可以幫助我們頁面中定制自定義邏輯的處理。有6個(gè)重要的事件,Init, Load, validate, event, render ,unload 簡稱 SILVER S – Start (does not signify anything as such just forms the word) , I – (Init) , L (Load) , V (Validate), E (Event) and R (Render)。
步驟4(M: HttpModule):一旦頁面對象是執(zhí)行和從內(nèi)存中卸載,httpmodule提供網(wǎng)頁執(zhí)行事件可以用來注入自定義處理邏輯。有4個(gè)重要的后處理事件postrequesthandlerexecute,releaserequeststate,updaterequestcache與endrequest。
事件詳細(xì)介紹:
部分 | Event事件 | Description描述 |
HttpModule | BeginRequest | 在 ASP.NET 響應(yīng)請求時(shí)作為 HTTP 執(zhí)行管線鏈中的***個(gè)事件發(fā)生。BeginRequest 事件發(fā)出信號表示創(chuàng)建任何給定的新請求。此事件始終被引發(fā),并且始終是請求處理期間發(fā)生的***個(gè)事件。 |
HttpModule | AuthenticateRequest | 當(dāng)安全模塊已建立用戶標(biāo)識時(shí)發(fā)生。AuthenticateRequest 事件發(fā)出信號表示配置的身份驗(yàn)證機(jī)制已對當(dāng)前請求進(jìn)行了身份驗(yàn)證。 |
HttpModule | AuthorizeRequest | 安全模塊已驗(yàn)證用戶授權(quán)時(shí)發(fā)生。AuthorizeRequest 事件發(fā)出信號表示 ASP.NET 已對當(dāng)前請求進(jìn)行了授權(quán)。 |
HttpModule | ResolveRequestCache | 在 ASP.NET 完成授權(quán)事件以使緩存模塊從緩存中為請求提供服務(wù)后發(fā)生,從而繞過事件處理程序(例如某個(gè)頁或 XML Web services)的執(zhí)行。 |
HttpModule | AcquireRequestState | 當(dāng) ASP.NET 獲取與當(dāng)前請求關(guān)聯(lián)的當(dāng)前狀態(tài)(如會話狀態(tài))時(shí)發(fā)生。AcquireRequestState 事件在創(chuàng)建了事件處理程序之后引發(fā)。 |
HttpModule | PreRequestHandlerExecute | 恰好在 ASP.NET 開始執(zhí)行事件處理程序(例如,某頁或某個(gè) XML Web services)前發(fā)生。 |
HttpHandler | ProcessRequest | Httphandler邏輯被執(zhí)行。 在本節(jié)中,我們會寫,需要將每頁擴(kuò)展執(zhí)行的邏輯。 |
Page | Init |
OnInit 方法執(zhí)行創(chuàng)建 Page 實(shí)例所需的初始化和設(shè)置步驟。在頁生命周期的此階段中,頁中聲明的服務(wù)器控件都已初始化為默認(rèn)狀態(tài);但每個(gè)控件的視圖狀態(tài)尚未填充。在 Page_Init 階段中,頁中的控件不能訪問該頁的其他服務(wù)器控件,無論其他控件是子控件還是父控件。不一定會創(chuàng)建其他服務(wù)器控件,也不一定能夠訪問它們 |
Page | Load | ASP.NET控件完成加載,你寫的UI操作邏輯或任何其他邏輯在這里。 |
Page | Validate | 如果你有你的頁面上有驗(yàn)證邏輯,這里檢查是否符合驗(yàn)證。 |
Render | 將頁面最終的輸出發(fā)送到瀏覽器,如果你想做出一些最終的HTML的變化,這是輸出到瀏覽器,你可以在這里輸入你的HTML的邏輯。 | |
Page | Unload | 頁面對象從內(nèi)存中卸載。 |
HttpModule | PostRequestHandlerExecute | 在 ASP.NET 事件處理程序(例如,某頁或某個(gè) XML Web service)執(zhí)行完畢時(shí)發(fā)生。 |
HttpModule | ReleaserequestState | 在 ASP.NET 執(zhí)行完所有請求事件處理程序后發(fā)生。該事件將使?fàn)顟B(tài)模塊保存當(dāng)前狀態(tài)數(shù)據(jù)。引發(fā) ReleaseRequestState 事件以后,應(yīng)用程序以該請求結(jié)束,并發(fā)出 ASP.NET 信號以存儲該請求狀態(tài)。 |
HttpModule | UpdateRequestCache | Before you end, if you want to update your cache.結(jié)束之前,如果您想更新您的高速緩存。 |
HttpModule | EndRequest | 當(dāng) ASP.NET 執(zhí)行完事件處理程序以使緩存模塊存儲將用于從緩存為后續(xù)請求提供服務(wù)的響應(yīng)時(shí)發(fā)生。 |
#p#
演示代碼:
在這段代碼中,我們創(chuàng)建HttpModule 和Httphandler 添加所有請求響應(yīng)的事件, 下面用 HttpModule Httphandler 跟蹤所有活動(dòng),并把它添加到一個(gè)全局性的集合變量中。
- public class clsHttpModule : IHttpModule
- {
- private HttpApplication httpApp;
- public static ArrayList objArrayList = new ArrayList();
- public clsHttpModule()
- {
- }
- public void Dispose()
- {
- }
- public void Init(HttpApplication context)
- {
- this.httpApp = context;
- httpApp.Context.Response.Clear();
- objArrayList.Clear();
- objArrayList.Add("httpModule:Init");
- httpApp.AuthenticateRequest += new EventHandler(OnAuthentication);
- httpApp.AuthorizeRequest += new EventHandler(OnAuthorization);
- httpApp.BeginRequest += new EventHandler(OnBeginrequest);
- httpApp.EndRequest += new EventHandler(OnEndRequest);
- httpApp.ResolveRequestCache += new EventHandler(OnResolveRequestCache);
- httpApp.AcquireRequestState += new EventHandler(OnAcquireRequestState);
- httpApp.PreRequestHandlerExecute += new EventHandler(OnPreRequestHandlerExecute);
- httpApp.PostRequestHandlerExecute += new EventHandler(OnPostRequestHandlerExecute);
- httpApp.ReleaseRequestState += new EventHandler(OnReleaseRequestState);
- httpApp.UpdateRequestCache += new EventHandler(OnUpdateRequestCache);
- }
- void OnUpdateRequestCache(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnUpdateRequestCache");
- }
- void OnReleaseRequestState(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnReleaseRequestState");
- }
- void OnPostRequestHandlerExecute(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnPostRequestHandlerExecute");
- }
- void OnPreRequestHandlerExecute(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnPreRequestHandlerExecute");
- }
- void OnAcquireRequestState(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnAcquireRequestState");
- }
- void OnResolveRequestCache(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnResolveRequestCache");
- }
- void OnAuthorization(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:OnAuthorization");
- }
- void OnAuthentication(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:AuthenticateRequest");
- }
- void OnBeginrequest(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:BeginRequest");
- }
- void OnEndRequest(object sender, EventArgs a)
- {
- objArrayList.Add("httpModule:EndRequest");
- objArrayList.Add("<hr>");
- foreach (string str in objArrayList)
- {
- httpApp.Context.Response.Write(str + "<br>");
- }
- }
- }
以下代碼片段是跟蹤 HttpHandler 的ProcessRequest '事件' :
- public class clsHttpHandler : IHttpHandler
- {
- public bool IsReusable
- {
- get { return true; }
- }
- public void ProcessRequest(HttpContext context)
- {
- clsHttpModule.objArrayList.Add("HttpHandler:ProcessRequest");
- context.Response.Redirect("~/Default.aspx");
- }
- }
頁面Page事件
- public partial class _Default : System.Web.UI.Page
- {
- protected void Page_init(object sender, EventArgs e)
- {
- clsHttpModule.objArrayList.Add("Page:Init");
- }
- protected void Page_Load(object sender, EventArgs e)
- {
- clsHttpModule.objArrayList.Add("Page:Load");
- }
- public override void Validate()
- {
- clsHttpModule.objArrayList.Add("Page:Validate");
- }
- protected void Button1_Click(object sender, EventArgs e)
- {
- clsHttpModule.objArrayList.Add("Page:Event");
- }
- protected override void Render(HtmlTextWriter output)
- {
- clsHttpModule.objArrayList.Add("Page:Render");
- base.Render(output);
- }
- protected void Page_Unload(object sender, EventArgs e)
- {
- clsHttpModule.objArrayList.Add("Page:UnLoad");
- }
- }
webconfig中配置:
- <httpModules>
- <add name="clsHttpModule" type="WebDemo1.Common.clsHttpModule"/>
- </httpModules>
頁面***次加載效果:
點(diǎn)擊button按鈕后:
#p#
HttpModule
HttpModule通過在某些事件中注冊,把自己插入ASP.NET請求處理管道。當(dāng)這些事件發(fā)生的時(shí)候,ASP.NET調(diào)用對相應(yīng)的HTTP模塊,這樣該模塊就能處理請求了。
常用操作:
1、向每個(gè)頁面動(dòng)態(tài)添加一些備注或說明性的文字:
2、判斷用戶登錄
如果定義了多個(gè)HttpModule,在web.config文件中引入自定義HttpModule的順序就決定了多個(gè)自定義HttpModule在處理一個(gè)HTTP請求的接管順序。
HttpHandler
HttpHandler是HTTP請求的處理中心,真正地對客戶端請求的服務(wù)器頁面做出編譯和執(zhí)行,并將處理過后的信息附加在HTTP請求信息流中再次返回到HttpModule中。
HttpHandler與HttpModule不同,一旦定義了自己的HttpHandler類,那么它對系統(tǒng)的HttpHandler的關(guān)系將是“覆蓋”關(guān)系。
為了驗(yàn)證請求是否進(jìn)入HttpHandler ,我們可以在WebConfig中 加入:
- <add verb="*" path="*.aspx" type="WebDemo1.Common.clsHttpHandler,WebDemo1"/>
調(diào)試程序看看。
綜上我們可以總計(jì)出一張圖:
ASP.NET頁面事件
在上面的部分,我們已經(jīng)看到了一個(gè)ASP.NET頁面完整的請求事件。 最重要的部分之一,是ASP.NET頁面,我們沒有詳細(xì)討論相同。我們詳細(xì)討論下ASP.NET的頁面事件。任何ASP.NET頁面中有2個(gè)部分 如圖:
注意 :大部分的開發(fā)者直接使用page_load方法的一切,這不是一個(gè)好的做法。例如 填充控件,設(shè)置視圖狀態(tài),應(yīng)用主題等,這些都是發(fā)生在頁面上的負(fù)荷。因此,我們可以在適當(dāng)?shù)氖录砑舆m當(dāng)?shù)倪壿?,這將真正使你的代碼整潔 符合邏輯。
序號 | 事件 | 控件初始化 | 狀態(tài) 是否 可用 |
表單數(shù)據(jù)是否 可用 |
什么邏輯可以寫在這里? |
1 | Init | No | No | No |
OnInit 方法執(zhí)行創(chuàng)建 Page 實(shí)例所需的初始化和設(shè)置步驟。在頁生命周期的此階段中,頁中聲明的服務(wù)器控件都已初始化為默認(rèn)狀態(tài);但每個(gè)控件的視圖狀態(tài)尚未填充。在 Page_Init 階段中,頁中的控件不能訪問該頁的其他服務(wù)器控件,無論其他控件是子控件還是父控件。不一定會創(chuàng)建其他服務(wù)器控件,也不一定能夠訪問它們。 |
2 | Load view state | Not guaranteed | Yes | Not guaranteed | 您可以訪問視圖狀態(tài)和任何同步邏輯 |
3 | PostBackdata | Not guaranteed | Yes | Yes | 你可以訪問表單數(shù)據(jù)。 |
4 | Load | Yes | Yes | Yes | 綁定控件 等等。。。 |
5 | Validate | Yes | Yes | Yes | 如果你的頁面有驗(yàn)證,或者你想為您的網(wǎng)頁執(zhí)行驗(yàn)證。 |
6 | Event | Yes | Yes | Yes | 如果這是通過點(diǎn)擊一個(gè)按鈕或下拉的變化,那么相對應(yīng)的事件將被執(zhí)行。 |
7 | Pre-render | Yes | Yes | Yes | 如果你想在這些控件視圖狀態(tài)***保存之前修改UI對象的結(jié)構(gòu)或?qū)傩灾怠?/td> |
8 | Save view state | Yes | Yes | Yes | 一旦所有服務(wù)器控件的變化完成,這一事件控制數(shù)據(jù)保存在視圖狀態(tài)。 |
9 | Render | Yes | Yes | Yes | 如果你想添加一些自定義的HTML輸出。 |
10 | Unload | Yes | Yes | Yes |
你可以在這里做任何清理。 |
原文鏈接:http://www.cnblogs.com/suizhouqiwei/archive/2012/08/15/2637775.html
【編輯推薦】