1、概述 (引自 微軟)
web
ASP.NET 頁運行時,此頁將經歷一個生命週期,在生命週期中將執行一系列處理步驟。這些步驟包括初始化、實例化控件、還原和維護狀態、運行事件處理程序代碼以及進行呈現。瞭解頁生命週期很是重要,由於這樣作您就能在生命週期的合適階段編寫代碼,以達到預期效果。此外,若是您要開發自定義控件,就必須熟悉頁生命週期,以便正確進行控件初始化,使用視圖狀態數據填充控件屬性以及運行任何控件行爲代碼。(控件的生命週期基於頁的生命週期,可是頁引起的控件事件比單獨的 ASP.NET 頁中可用的事件多。)sql
通常來講,頁要經歷下表概述的各個階段。除了頁生命週期階段之外,在請求先後還存在應用程序階段,可是這些階段並不特定於頁。有關更多信息,請參見 IIS 5.0 和 6.0 的 ASP.NET 應用程序生命週期概述。數據庫
階段express |
說明編程 |
---|---|
頁請求windows |
頁請求發生在頁生命週期開始以前。用戶請求頁時,ASP.NET 將肯定是否須要分析和編譯頁(從而開始頁的生命週期),或者是否能夠在不運行頁的狀況下發送頁的緩存版本以進行響應。瀏覽器 |
開始緩存 |
在開始階段,將設置頁屬性,如 Request 和 Response。在此階段,頁還將肯定請求是回發請求仍是新請求,並設置 IsPostBack 屬性。此外,在開始階段期間,還將設置頁的 UICulture 屬性。服務器 |
頁初始化session |
頁初始化期間,可使用頁中的控件,並將設置每一個控件的 UniqueID 屬性。此外,任何主題都將應用於頁。若是當前請求是回發請求,則回發數據還沒有加載,而且控件屬性值還沒有還原爲視圖狀態中的值。 |
加載 |
加載期間,若是當前請求是回發請求,則將使用從視圖狀態和控件狀態恢復的信息加載控件屬性。 |
驗證 |
在驗證期間,將調用全部驗證程序控件的 Validate 方法,此方法將設置各個驗證程序控件和頁的 IsValid 屬性。 |
回發事件處理 |
若是請求是回發請求,則將調用全部事件處理程序。 |
呈現 |
在呈現以前,會針對該頁和全部控件保存視圖狀態。在呈現階段中,頁會針對每一個控件調用 Render 方法,它會提供一個文本編寫器,用於將控件的輸出寫入頁的 Response 屬性的 OutputStream 中。 |
卸載 |
徹底呈現頁並已將頁發送至客戶端、準備丟棄該頁後,將調用卸載。此時,將卸載頁屬性(如 Response 和 Request)並執行清理。 |
在頁生命週期的每一個階段中,頁將引起可運行您本身的代碼進行處理的事件。對於控件事件,經過以聲明方式使用屬性(如 onclick)或以使用代碼的方式,都可將事件處理程序綁定到事件。
頁還支持自動事件鏈接,即,ASP.NET 將查找具備特定名稱的方法,並在引起了特定事件時自動運行這些方法。若是 @ Page 指令的 AutoEventWireup 屬性設置爲 true(或者未定義該屬性,由於該屬性默認爲 true),頁事件將自動綁定至使用 Page_事件的命名約定的方法(如 Page_Load 和 Page_Init)。有關自動事件鏈接的更多信息,請參見 ASP.NET Web 服務器控件事件模型。
下表列出了最經常使用的頁生命週期事件。除了列出的事件外還有其餘事件;不過,大多數頁處理方案不使用這些事件。而是主要由 ASP.NET 網頁上的服務器控件使用,以初始化和呈現它們自己。若是要編寫本身的 ASP.NET 服務器控件,則須要詳細瞭解這些階段。有關建立自定義控件的信息,請參見開發自定義 ASP.NET 服務器控件。
頁事件 |
典型使用 |
---|---|
使用該事件來執行下列操做:
|
|
在全部控件都已初始化且已應用全部外觀設置後引起。使用該事件來讀取或初始化控件屬性。 |
|
由 Page 對象引起。使用該事件來處理要求先完成全部初始化工做的任務。 |
|
若是須要在 Load 事件以前對頁或控件執行處理,請使用該事件。 |
|
Page 在 Page 上調用 OnLoad 事件方法,而後以遞歸方式對每一個子控件執行相同操做,如此循環往復,直到加載完本頁和全部控件爲止。 使用 OnLoad 事件方法來設置控件中的屬性並創建數據庫鏈接。 |
|
控件事件 |
使用這些事件來處理特定控件事件,如 Button 控件的 Click 事件或 TextBox 控件的 TextChanged 事件。 |
對須要加載頁上的全部其餘控件的任務使用該事件。 |
|
在該事件發生前:
頁上的每一個控件都會發生 PreRender 事件。使用該事件對頁或其控件的內容進行最後更改。 |
|
在該事件發生前,已針對頁和全部控件保存了 ViewState。將忽略此時對頁或控件進行的任何更改。 使用該事件執行知足如下條件的任務:要求已經保存了視圖狀態,但未對控件進行任何更改。 |
|
這不是事件;在處理的這個階段,Page 對象會在每一個控件上調用此方法。全部 ASP.NET Web 服務器控件都有一個用於寫出發送給瀏覽器的控件標記的 Render 方法。 若是建立自定義控件,一般要覆蓋此方法以輸出控件的標記。不過,若是自定義控件只合並標準的 ASP.NET Web 服務器控件,不合並自定義標記,則不須要覆蓋 Render 方法。有關更多信息,請參見開發自定義 ASP.NET 服務器控件。 用戶控件(.ascx 文件)自動合併呈現,所以不須要在代碼中顯式呈現該控件。 |
|
該事件首先針對每一個控件發生,繼而針對該頁發生。在控件中,使用該事件對特定控件執行最後清理,如關閉控件特定數據庫鏈接。 對於頁自身,使用該事件來執行最後清理工做,如:關閉打開的文件和數據庫鏈接,或完成日誌記錄或其餘請求特定任務。
說明:
在卸載階段,頁及其控件已被呈現,所以沒法對響應流作進一步更改。若是嘗試調用方法(如 Response.Write 方法),則該頁將引起異常。
|
各個 ASP.NET 服務器控件都有本身的生命週期,該生命週期與頁生命週期相似。例如,控件的 Init 和 Load 事件在相應的頁事件期間發生。
雖然 Init 和 Load 都在每一個控件上以遞歸方式發生,但它們的發生順序相反。每一個子控件的 Init 事件(還有 Unload 事件)在爲其容器引起相應的事件以前發生(由下到上)。可是,容器的 Load 事件是在其子控件的 Load 事件以前發生(由上到下)。
能夠經過處理控件的事件(如 Button 控件的 Click 事件和 ListBox 控件的 SelectedIndexChanged 事件)來自定義控件的外觀或內容。在某些狀況下,可能也需處理控件的 DataBinding 或 DataBound 事件。有關更多信息,請參見各個控件的類參考主題以及開發自定義 ASP.NET 服務器控件。
當從 Page 類繼承類時,除了能夠處理由頁引起的事件之外,還能夠覆蓋頁的基類中的方法。例如,能夠覆蓋頁的 InitializeCulture 方法,以便動態設置區域性信息。注意,在使用 Page_事件語法建立事件處理程序時,將隱式調用基實現,所以無需在方法中調用它。例如,不管是否建立 Page_Load 方法,始終都會調用頁基類的 OnLoad 方法。可是,若是使用 override 關鍵字(在 Visual Basic 中爲 Overrides)覆蓋頁的 OnLoad 方法,則必須顯式調用基方法。例如,若是在頁中覆蓋 OnLoad 方法,則必須調用 base.Load(在 Visual Basic 中爲 MyBase.Load)以運行基實現。
若是控件是在運行時動態建立的,或者是以聲明方式在數據綁定控件的模板中建立的,它們的事件最初與頁上的其餘控件的事件並不一樣步。例如,對於運行時添加的控件,Init 和 Load 事件在頁生命週期中的發生時間可能要比以聲明方式建立的控件的相同事件晚得多。所以,從實例化那一刻起,動態添加的控件的事件就一直是在模板中的控件的事件以後發生,直到遇上該控件加入 Controls 集合時所對應事件爲止。
通常來講,除非存在嵌套數據綁定控件,不然,您沒必要擔憂這種狀況。若是子控件已執行數據綁定,但其容器控件還沒有執行數據綁定,則子控件中的數據與其容器控件中的數據可能不一樣步。若是子控件中的數據根據容器控件中的數據綁定值執行了處理,這種狀況則尤爲顯著。
例如,假定有一個 GridView,它的每一行顯示一條公司記錄,此外,有一個 ListBox 控件包含公司管理者列表。若要填充管理者列表,則須要將 ListBox 控件綁定到一個數據源控件(如 SqlDataSource),後者在查詢中使用 CompanyID 來檢索公司管理者數據。
若是以聲明方式設置了 ListBox 控件的數據綁定屬性(如 DataSourceID 和 DataMember),ListBox 控件將嘗試在包含行的 DataBinding 事件期間綁定到其數據源。不過,行的 CompanyID 字段直到 GridView 控件的 RowDataBound 事件發生後才包含值。這種狀況下,先綁定子控件(ListBox 控件),後綁定包含控件(GridView 控件),所以它們的數據綁定階段並不一樣步。
若要避免此種狀況,須要將 ListBox 控件的數據源控件與 ListBox 控件自身放在同一模板項中,而且不要以聲明方式設置 ListBox 的數據綁定屬性。而應在 RowDataBound 事件期間在運行時以編程方式設置它們,這樣,到 CompanyID 信息可用時 ListBox 控件纔會綁定到其數據。
有關更多信息,請參見使用數據源控件綁定到數據。
爲了幫助您理解頁生命週期與數據綁定事件之間的關係,下表列出了數據綁定控件(如 GridView、DetailsView 和 FormView 控件)中與數據相關的事件。
控件事件 |
典型使用 |
---|---|
該事件在包含控件(或 Page 對象)的 PreRender 事件以前由數據綁定控件引起,會標記控件到數據的綁定過程的起點。 若是須要,使用該事件以手動方式打開數據庫鏈接。(數據源控件一般不須要如此操做。) |
|
RowCreated(僅限 GridView)或 ItemCreated(DataList、DetailsView、SiteMapPath、DataGrid、FormView、Repeater 和 ListView 控件) |
使用該事件來操做不依賴於數據綁定的內容。例如,在運行時,能夠以編程方式向 GridView 控件中的頁眉或頁腳行添加格式。 |
RowDataBound(僅限 GridView)或 ItemDataBound(DataList、SiteMapPath、DataGrid、Repeater 和 ListView 控件) |
當該事件發生時,行或項中的數據可用,所以,能夠在子數據源控件上格式化數據或設置 FilterExpression 屬性,以便顯示行或項中的相關數據。 |
該事件在數據綁定控件中標記數據綁定操做的結尾。在 GridView 控件中,會針對全部行和任何子控件完成數據綁定。 使用該事件格式化數據綁定內容,或在依賴來自當前控件的內容的值的其餘控件中啓動數據綁定。(有關詳細信息,請參見本主題中前面的「添加的控件的追趕事件」。) |
Login 控件可使用 Web.config 文件中的設置來自動管理成員資格驗證。不過,若是應用程序要求您自定義控件的工做方式,或者您要了解 Login 控件事件與頁生命週期的關聯方式,可使用下表中列出的事件。
控件事件 |
典型使用 |
---|---|
在回發期間,當頁的 LoadComplete 事件發生後就會引起該事件。它標記登陸過程的起點。 對必須在驗證過程開始前發生的任務使用該事件。 |
|
該事件在 LoggingIn 事件以後引起。 使用該事件來覆蓋或加強 Login 控件的默認驗證行爲。 |
|
該事件在驗證用戶名和密碼後引起。 使用該事件來重定向到另外一個頁或動態設置控件中的文本。若是出現錯誤或驗證失敗,就不會發生該事件。 |
|
若是驗證失敗,將引起該事件。 使用該事件來設置控件中的問題解釋文本或將用戶定向到不一樣的頁。 |
2、實例講解
一、運行page頁面時的執行順序
using System; using System.Data; using System.Configuration; using System.Web; using System.Web.Security; using System.Web.UI; using System.Web.UI.WebControls; using System.Web.UI.WebControls.WebParts; using System.Web.UI.HtmlControls; public partial class _Default : Page { protected void Page_Load(object sender, EventArgs e) { } #region OnPreInit 第一步 protected override void OnPreInit(EventArgs e) { //檢查 IsPostBack 屬性來肯定是否是第一次處理該頁。 //建立或從新建立動態控件。 //動態設置主控頁。 //動態設置 Theme 屬性。 //讀取或設置配置文件屬性值。 //注意 //若是請求是回發請求,則控件的值還沒有從視圖狀態還原。若是在此階段設置控件屬性,則其值可能會在下一事件中被重寫。 base.OnPreInit(e); } #endregion #region OnInit 第二步 protected override void OnInit(EventArgs e) { //在全部控件都已初始化且已應用全部外觀設置後引起。使用該事件來讀取或初始化控件屬性。 base.OnInit(e); } #endregion #region OnInitComplete 第三步 protected override void OnInitComplete(EventArgs e) { //由 Page 對象引起。使用該事件來處理要求先完成全部初始化工做的任務。 base.OnInitComplete(e); } #endregion #region PreLoad 第四步 protected override void OnPreLoad(EventArgs e) { //若是須要在 Load 事件以前對頁或控件執行處理,請使用該事件。 //在 Page 引起該事件後,它會爲自身和全部控件加載視圖狀態,而後會處理 Request 實例包括的任何回發數據。 base.OnPreLoad(e); } #endregion #region OnLoad 第五步 protected override void OnLoad(EventArgs e) { //Page 在 Page 上調用 OnLoad 事件方法,而後以遞歸方式對每一個子控件執行相同操做,如此循環往復,直到加載完本頁和全部控件爲止。 //使用 OnLoad 事件方法來設置控件中的屬性並創建數據庫鏈接。 base.OnLoad(e); } #endregion #region 控件事件 第六步 protected void Button1_Click(object sender, EventArgs e) { //用這些事件來處理特定控件事件,如 Button 控件的 Click 事件或 TextBox 控件的 TextChanged 事件。 //注意 //在回發請求中,若是頁包含驗證程序控件,請在執行任何處理以前檢查 Page 和各個驗證控件的 IsValid 屬性。 } #endregion #region OnLoadComplete 第七步 protected override void OnLoadComplete(EventArgs e) { //對須要加載頁上的全部其餘控件的任務使用該事件。 base.OnLoadComplete(e); } #endregion #region OnPreRender 第八步 protected override void OnPreRender(EventArgs e) { //在該事件發生前: //Page 對象會針對每一個控件和頁調用 EnsureChildControls。 //設置了 DataSourceID 屬性的每一個數據綁定控件會調用 DataBind 方法。有關更多信息,請參見下面的數據綁定控件的數據綁定事件。 //頁上的每一個控件都會發生 PreRender 事件。使用該事件對頁或其控件的內容進行最後更改。 base.OnPreRender(e); } #endregion #region SaveStateComplete 第九步 protected override void OnSaveStateComplete(EventArgs e) { //在該事件發生前,已針對頁和全部控件保存了 ViewState。將忽略此時對頁或控件進行的任何更改。 //使用該事件執行知足如下條件的任務:要求已經保存了視圖狀態,但未對控件進行任何更改。 base.OnSaveStateComplete(e); } #endregion #region Render 第十步 //Render //這不是事件;在處理的這個階段,Page 對象會在每一個控件上調用此方法。全部 ASP.NET Web 服務器控件都有一個用於寫出發送給瀏覽器的控件標記的 Render 方法。 //若是建立自定義控件,一般要重寫此方法以輸出控件的標記。不過,若是自定義控件只合並標準的 ASP.NET Web 服務器控件,不合並自定義標記,則不須要重寫 Render 方法。有關更多信息,請參見開發自定義 ASP.NET 服務器控件。 //用戶控件(.ascx 文件)自動合併呈現,所以不須要在代碼中顯式呈現該控件。 #endregion #region OnUnload 第十一步 protected override void OnUnload(EventArgs e) { //該事件首先針對每一個控件發生,繼而針對該頁發生。在控件中,使用該事件對特定控件執行最後清理,如關閉控件特定數據庫鏈接。 //對於頁自身,使用該事件來執行最後清理工做,如:關閉打開的文件和數據庫鏈接,或完成日誌記錄或其餘請求特定任務。 //注意: //在卸載階段,頁及其控件已被呈現,所以沒法對響應流作進一步更改。若是嘗試調用方法(如 Response.Write 方法),則該頁將引起異常。 base.OnUnload(e); } #endregion }
當頁面進行回發時,如點擊按鈕,以上事件都會從新執行一次,這時的執行順序爲:
1. OnPreInit
2. OnInit
3. OnInitComplete
4. OnPreLoad
5. Page_Load
6. OnLoad
7. Button_Click
8. OnLoadComplete
9. OnPreRender
能夠看到,Button_Click事件位於OnLoad以後執行,能夠測試一下:
public partial class TestControls : System.Web.UI.Page { static int count = 0; protected void Page_Load(object sender, EventArgs e) { Response.Write(count+ "Page_Load <br />"); count++; } protected override void OnPreInit(EventArgs e) { base.OnPreInit(e); Response.Write(count + "OnPreInit <br />"); count++; } protected override void OnInit(EventArgs e) { base.OnInit(e); Response.Write(count + "OnInit <br />"); count++; } protected override void OnLoad(EventArgs e) { base.OnLoad(e); Response.Write(count + "OnLoad <br />"); count++; } protected override void OnPreLoad(EventArgs e) { base.OnPreLoad(e); Response.Write(count + "OnPreLoad <br />"); count++; } protected override void OnLoadComplete(EventArgs e) { base.OnLoadComplete(e); Response.Write(count + "OnLoadComplete <br />"); count++; } protected override void OnInitComplete(EventArgs e) { base.OnInitComplete(e); Response.Write(count + "OnInitComplete <br />"); count++; } protected override void OnUnload(EventArgs e) { base.OnUnload(e); } protected override void OnDataBinding(EventArgs e) { base.OnDataBinding(e); Response.Write(count + "OnDataBinding <br />"); count++; } protected override void OnPreRender(EventArgs e) { base.OnPreRender(e); Response.Write(count + "OnPreRender <br />"); count++; } protected void btnGraphics_Click(object sender, EventArgs e) { //Bitmap bmp = new Bitmap(10, 10); //Graphics g = Graphics.FromImage(bmp); Response.Write(count + "btnGraphics_Click <br />"); count++; } }
二、熟悉請求管道實現程序運行的全過程:
(1):BeginRequest: 開始處理請求
(2):AuthenticateRequest受權驗證請求,獲取用戶受權信息
(3):PostAuthenticateRequest獲取成功
(4): AunthorizeRequest 受權,通常來檢查用戶是否得到權限
(5):PostAuthorizeRequest:得到受權
(6):ResolveRequestCache:獲取頁面緩存結果
(7):PostResolveRequestCache 已獲取緩存
(8):PostMapRequestHandler 建立頁面對象
(9):AcquireRequestState 獲取Session-----先判斷當前頁面對象是否實現了IRequiresSessionState接口,若是實現了,則從瀏覽器發來的請求報文體中得到SessionID,併到服務器的Session池中得到對應的Session對象,最後賦值給HttpContext的Session屬性
(10)PostAcquireRequestState 得到Session
(11)PreRequestHandlerExecute:準備執行頁面對象
執行頁面對象的ProcessRequest方法
(12)PostRequestHandlerExecute 執行完頁面對象了
(13)ReleaseRequestState 釋放請求狀態
(14)PostReleaseRequestState 已釋放請求狀態
(15)UpdateRequestCache 更新緩存
(16)PostUpdateRequestCache 已更新緩存
(17)LogRequest 日誌記錄
(18)PostLogRequest 已完成日誌
(19)EndRequest 完成
public class getsession : System.Web.UI.Page, IReadOnlySessionState { string ss = ""; public void Init(HttpApplication context) { //這裏能夠根據需求,添加各個請求管道 //獲取Session context.AcquireRequestState += new EventHandler(context_AcquireRequestState); //獲取Url context.BeginRequest += new EventHandler(context_BeginRequest); } void context_AcquireRequestState(object sender, EventArgs e) { if (Session["user"] != null) { ss = Session["user"].ToString(); } } void context_BeginRequest(object sender, EventArgs e) { //得到當前頁面請求管道的HttpApplication對象 HttpApplication application = sender as HttpApplication; HttpContext context = application.Context;//得到上下文對象 string url = context.Request.Url.LocalPath;//得到URL(不包含域名和路徑) } }