常規頁生命周期階段
--------------------------------------------------------------------------------
一般來說,頁要經歷下表概述的各個階段。除了頁生命周期階段以外,在請求前後還存在應用程序階段,但是這些階段並不特定於頁。 階段
說明
頁請求
頁請求發生在頁生命周期開始之前。用戶請求頁時,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 網頁上的服務器控件使用,以初始化和呈現它們本身。如果要編寫自己的 ASP.NET 服務器控件,則需要詳細了解這些階段。
頁事件
典型使用
PreInit
使用該事件來執行下列操作:
· 檢查 IsPostBack 屬性來確定是不是第一次處理該頁。
· 創建或重新創建動態控件。
· 動態設置主控頁。
· 動態設置 Theme 屬性。
· 讀取或設置配置文件屬性值。
Init
在所有控件都已初始化且已應用所有外觀設置後引發。使用該事件來讀取或初始化控件屬性。
InitComplete
由 Page 對象引發。使用該事件來處理要求先完成所有初始化工作的任務。
PreLoad
如果需要在 Load 事件之前對頁或控件執行處理,請使用該事件。
在 Page 引發該事件後,它會為自身和所有控件加載視圖狀態,然後會處理 Request 實例包括的任何回發數據。
Load
Page 在 Page 上調用 OnLoad 事件方法,然後以遞歸方式對每個子控件執行相同操作,如此循環往復,直到加載完本頁和所有控件為止。
使用 OnLoad 事件方法來設置控件中的屬性並建立數據庫連接。
控件事件
使用這些事件來處理特定控件事件,如 Button 控件的 Click 事件或 TextBox 控件的 TextChanged 事件。
LoadComplete
對需要加載頁上的所有其他控件的任務使用該事件。
PreRender
在該事件發生前:
· Page 對象會針對每個控件和頁調用 EnsureChildControls。
· 設置了 DataSourceID 屬性的每個數據綁定控件會調用 DataBind 方法。有關更多信息,請參見本主題後面的數據綁定控件的數據綁定事件。
頁上的每個控件都會發生 PreRender 事件。使用該事件對頁或其控件的內容進行最後更改。
SaveStateComplete
在該事件發生前,已針對頁和所有控件保存了 ViewState。將忽略此時對頁或控件進行的任何更改。
使用該事件執行滿足以下條件的任務:要求已經保存了視圖狀態,但未對控件進行任何更改。
Render
這不是事件;在處理的這個階段,Page 對象會在每個控件上調用此方法。所有 ASP.NET Web 服務器控件都有一個用於寫出發送給浏覽器的控件標記的 Render 方法。
如果創建自定義控件,通常要覆蓋此方法以輸出控件的標記。不過,如果自定義控件只合並標准的 ASP.NET Web 服務器控件,不合並自定義標記,則不需要覆蓋 Render 方法。有關更多信息,請參見開發自定義 ASP.NET 服務器控件。
用戶控件(.ascx 文件)自動合並呈現,因此不需要在代碼中顯式呈現該控件。
Unload
該事件首先針對每個控件發生,繼而針對該頁發生。在控件中,使用該事件對特定控件執行最後清理,如關閉控件特定數據庫連接。
對於頁自身,使用該事件來執行最後清理工作,如:關閉打開的文件和數據庫連接,或完成日志記錄或其他請求特定任務。
頁面經歷了如下階段【其中一些階段標記為僅僅回傳,是說只有在被回傳到服務器時,頁面才經歷這些階段】:
01:頁面首先從QueryString或者Request對象的Form集合中獲得回傳數據。
02:頁面檢測回傳數據集合(NameValueCollection,Form或者QueryString)是否包含一個鍵為_CallBackId的項。如 果存在,那麼設置其Boolean屬性IsCallBack為True,以便通過AspNet客戶端回調機制,標明頁面已經回傳到服務器。
03:預初始化(PreInit):
在頁面生命周期的預初始化階段執行如下操作:
a:調用OnPreInit方法引發PreInit事件。
b:利用App_Themes目錄中的內容初始化主題,以動態實現一個PageTheme類型的類,
編譯該類,並創建一個編譯類的實例,接著將實例賦值給它的PageTheme屬性值
c:應用母版頁
04:初始化(Init):
在頁面生命周期的初始化階段執行以下操作
a:遞歸初始化Controls集合中的控件。初始化包括設置這些控件的屬性,
例如:Page,Id和NameContainer等
b:遞歸應用控件皮膚
c:調用OnInit方法以引發自身的Init事件,接著遞歸調用子控件的OnInit方法來引發它們的Init事件
d:調用自身的TrackViewState方法來啟動自身的視圖狀態跟蹤,接著遞歸調用子控件
的TrackViewState方法來啟動它們的視圖狀態跟蹤。
05:完成初始化(InitComplete):
頁面調用OnInitComplete方法來引發InitComplete事件。該事件標明了初始化階段的結束。
此時,頁面Controls集合的所有控件都被初始化了。
06:加載控件狀態(Load Control State)[PostBack Only]:
頁面遞歸調用Control集合中控件的LoadControlState方法,這些控件已經調用了Page類
的RegisterRequiresControlState方法,以使用它們的控件狀態。
07:加載視圖狀態(Load View State)[PostBack Only]:
頁面首先調用自身的LoadViewState方法,接著遞歸調用Controls集合控件的LoadViewState方法,以允許它們加載它們的已經保存的視圖狀態。
08:加載回傳數據(Load Post Data)[PostBack Only]:
頁面調用實現IPostBackDataHandler接口的控件的LoadPostData方法,並將回傳數據傳遞給該方法。每個控件的LoadPostDate方法都必須訪問回傳數據,並據此更新相應的控件屬性。
例如:TextBox控件的LoadPostData方法將文本框的新值設置為TextBox控件的Text屬性值。
09:預加載(PreLoad):
頁面調用OnPreLoad方法以引發PreLoad事件。該事件表示頁面生命周期將進入加載階段。
10:加載(Load):
頁面首先調用自身的OnLoad方法以引發自身的Load事件,接著遞歸調用Controls集合中控件的OnLoad方法以引發它們的Load事件。頁面開發人員可以為Load事件注冊回調,那麼就可以通過編程將子控件添加到頁面的Controls集合中。
11:加載回傳數據(Load Post Data)[PostBack Only Second Try]:
頁面調用控件的LoadPostBack方法。這些控件如果實現了IPostBackDataHandler接口,那麼在加載階段,它們已通過編程添加到Controls集合中。
12:引發回傳數據修改事件(Raise Post Data Changed Event)[PostBack Only]:
頁面調用控件的RaisePostDataChangeEvent方法,這些控件的LoadPostData方法返回true。
RaisePostDataChangeEvent方法引發了回傳數據修改事件。例如:當文本框的新值與舊值
不同時,那麼TextBox控件將引發該事件。
13:引發回傳事件(Raise PostBack Event)[PostBack Only]: