ASP.NET 頁運行時,此頁將經歷一個生命周期,在生命周期中將執行一系列處理步驟。這些步驟包括初始化、實例化控件、還原和維護狀態、運行事件處理程序代碼以及進行呈現。了解頁生命周期非常重要,因為這樣做您就能在生命周期的合適階段編寫代碼,以達到預期效果。此外,如果您要開發自定義控件,就必須熟悉頁生命周期,以便正確進行控件初始化,使用視圖狀態數據填充控件屬性以及運行任何控件行為代碼。(控件的生命周期基於頁的生命周期,但是頁引發的控件事件比單獨的 ASP.Net 頁中可用的事件多。)
常規頁生命周期階段
一般來說,頁要經歷下表概述的各個階段。除了頁生命周期階段以外,在請求前後還存在應用程序階段,但是這些階段並不特定於頁。有關更多信息,請參見 ASP.Net 應用程序生命周期概述。
階段
說明
頁請求
頁請求發生在頁生命周期開始之前。用戶請求頁時,ASP.Net 將確定是否需要分析和編譯頁(從而開始頁的生命周期),或者是否可以在不運行頁的情況下發送頁的緩存版本以進行響應。
開始
在開始階段,將設置頁屬性,如 Request 和 Response。在此階段,頁還將確定請求是回發請求還是新請求,並設置 IsPostBack 屬性。此外,在開始階段期間,還將設置頁的 UICulture 屬性。
頁初始化
頁初始化期間,可以使用頁中的控件,並將設置每個控件的 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 網頁上的服務器控件使用,以初始化和呈現它們本身。
頁事件
典型使用
PreInit
使用該事件來執行下列操作:
· 檢查 IsPostBack 屬性來確定是不是第一次處理該頁。
· 創建或重新創建動態控件。
· 動態設置主控頁。
· 動態設置 Theme 屬性。
· 讀取或設置配置文件屬性值。
如果請求是回發請求,則控件的值尚未從視圖狀態還原。如果在此階段設置控件屬性,則其值可能會在下一事件中被重寫。
·
Init
在所有控件都已初始化且已應用所有外觀設置後引發。使用該事件來讀取或初始化控件屬性。
InitComplete
由 Page 對象引發。使用該事件來處理要求先完成所有初始化工作的任務。
PreLoad
如果需要在 Load 事件之前對頁或控件執行處理,請使用該事件。
在 Page 引發該事件後,它會為自身和所有控件加載視圖狀態,然後會處理 Request 實例包括的任何回發數據。
Load
Page 在 Page 上調用 OnLoad 事件方法,然後以遞歸方式對每個子控件執行相同操作,如此循環往復,直到加載完本頁和所有控件為止。
使用 OnLoad 事件方法來設置控件中的屬性並建立數據庫連接。
控件事件
使用這些事件來處理特定控件事件,如 Button 控件的 Click 事件或 TextBox 控件的 TextChanged 事件。
在回發請求中,如果頁包含驗證程序控件,請在執行任何處理之前檢查 Page 和各個驗證控件的 IsValid 屬性。
LoadComplete
對需要加載頁上的所有其他控件的任務使用該事件。
PreRender
在該事件發生前:
· Page 對象會針對每個控件和頁調用 EnsureChildControls。
· 設置了 DataSourceID 屬性的每個數據綁定控件會調用 DataBind 方法。有關更多信息,請參見下面的數據綁定控件的數據綁定事件。
頁上的每個控件都會發生 PreRender 事件。使用該事件對頁或其控件的內容進行最後更改。
SaveStateComplete
在該事件發生前,已針對頁和所有控件保存了 VIEwState。將忽略此時對頁或控件進行的任何更改。
使用該事件執行滿足以下條件的任務:要求已經保存了視圖狀態,但未對控件進行任何更改。
Render
這不是事件;在處理的這個階段,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 控件)中與數據相關的事件。
控件事件
典型使用
DataBinding
該事件在包含控件(或 Page 對象)的 PreRender 事件之前由數據綁定控件引發,會標記控件到數據的綁定過程的起點。
如果需要,使用該事件以手動方式打開數據庫連接。(數據源控件通常不需要如此操作。)
RowCreated(僅限 GridVIEw)或 ItemCreated(DataList、DetailsVIEw、SiteMapPath、DataGrid、FormVIEw 和 Repeater 控件)
使用該事件來操作不依賴於數據綁定的內容。例如,在運行時,可以以編程方式向 GridVIEw 控件中的頁眉或頁腳行添加格式。
RowDataBound(僅限 GridVIEw)或 ItemDataBound(DataList、SiteMapPath、DataGrid 和 Repeater 控件)
當該事件發生時,行或項中的數據可用,因此,可以在子數據源控件上格式化數據或設置 FilterExpression 屬性,以便顯示行或項中的相關數據。
DataBound
該事件在數據綁定控件中標記數據綁定操作的結尾。在 GridVIEw 控件中,會針對所有行和任何子控件完成數據綁定。
使用該事件格式化數據綁定內容,或在依賴來自當前控件的內容的值的其他控件中啟動數據綁定。(有關詳細信息,請參見本主題中前面的“添加的控件的追趕事件”。)