前言
只要對VIEwState稍有了解,就會知道,ASP.Net頁面中VIEwState一般是存儲在頁面的一個隱藏域中:
<input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE" value="一堆亂七八糟的東西">
當我們浏覽頁面源文件時,看到的那一大堆(特別是當頁面有個有大量數據的DataGrid,或在ASP.Net2.0中的GridView時)亂七八糟的東西的時候,那就是VIEwState了。
基礎知識
因為,在ASP.Net2.0中VIEwState的持久性存儲機制有了些新的變化,所以,還是簡單介紹下相關的東西。
在ASP.Net1.1中,只提供了頁面隱藏域的持久性機制,這樣在某些情況下不得不放棄使用ViewState,試想下,如果你的DataGrid中有上萬條記錄(別認為這種變態的需要是沒有的,有人就碰到過),如果啟用了VIEwState,你感保證你的IIS服務器能承受得住嗎,網絡承受得主嗎?當然你是可以通過重寫Page.SavePageStateToPersistenceMedium()方法來更改你的存儲機制,但別忘了重寫Page.LoadPageStateFromPersistenceMedium(),它們可是一對的啊。
ASP.NET2.0 中的默認視圖狀態持久性機制依然是在頁上的一個隱藏 Html 元素(一個 type 屬性設置為 "hidden" 的元素)中將狀態信息保留為一個 Base64 編碼的字符串。ASP.Net 頁使用 HiddenFIEldPageStatePersister 對象執行此項工作,並使用一個 IStateFormatter 實例對對象狀態信息進行序列化和反序列化。或者,對於帶寬和資源有限的移動客戶端,您也可以使用 SessionPageStatePersister 類在服務器上的 Session 對象中存儲頁的視圖狀態,其實也就多了個Session持久機制而已,讓我們把頁面狀態保存在Session中,而不是頁面中,這對帶寬是一種節省。
但你要深入的了解VIEwState持久機制的話,抽象類PageStatePersister你是應該去了解的,要在不能支持現有視圖狀態持久性機制的客戶端上保留視圖狀態,可以擴展 PageStatePersister 類,引入您自己的視圖狀態持久性方法,並且可以使用頁適配器將 ASP.Net 應用程序配置為根據為其提供頁的客戶端的類型使用不同的視圖狀態持久性機制。從 PageStatePersister 類派生的類必須重寫 Save 抽象方法,以便在持久性介質中存儲視圖狀態和控件狀態,同時重寫 Load 方法以提取狀態信息。如果需要將視圖狀態和控件狀態序列化為字符串,可以使用通過 StateFormatter 屬性來訪問的 IStateFormatter 對象。它可以高效地將對象狀態信息序列化和反序列化為 Base64 編碼字符串。還可以重寫 StateFormatter 屬性以提供自己的對象狀態序列化機制,如何為之,我的代碼中都有介紹,很簡單,看看就明白了。
VIEwState持久性機制
隱藏域
這個就不介紹了,默認的就是這種。就入前言中的那樣。
Session
在ASP.Net2.0中只要重寫PageStatePersister屬性就可以了。
protected override PageStatePersister PageStatePersister
{
get
{
return new SessionPageStatePersister(Page);
}
}
要是在ASP.Net1.1中需要重寫LoadPageStateFromPersistenceMedium這兩個方法:
protected override object LoadPageStateFromPersistenceMedium()
{
return Session["VIEwState"];
}
protected override void SavePageStateToPersistenceMedium(object vIEwState)
{
Session["ViewState"] = vIEwState;
RegisterHiddenField("__VIEWSTATE", "");
}
數據庫(我的示例是SQL Server2000)
在ASP1.1中,請注意下面紫色的那行,我也不太清楚那有什麼用,它讓我郁悶了好幾天,等下你就明白我的郁悶了。還有下面的代碼只是湊我的源碼中拷貝出來的,你完全可以不這樣寫的,除了那些必要的外。
protected override void SavePageStateToPersistenceMedium(object state)
{
string viewStateID = "VIEWSTATE#" + Session.SessionID.ToString() + "#" + DateTime.Now.Ticks.ToString();
ClientScript.RegisterHiddenField("__VIEWSTATE_KEY", vIEwStateID);
ClientScript.RegisterHiddenField("__VIEWSTATE","");//請注意
try
{
if (losFormatter == null)
{
losFormatter = new LosFormatter();
}
StringWriter sw = new StringWriter();
losFormatter.Serialize(sw, state);
Common.ViewStateData vsd = new VIEwStateData();
vsd.ViewStateID = vIEwStateID;
vsd.VIEwState = sw.ToString();
da = new DataAccess();
string error = da.SaveVIEwState(vsd);
Response.Write(error);
}
catch (Exception ex)
{
Response.Write(ex.Message
);
}
}
protected override object LoadPageStateFromPersistenceMedium()
{
string vIEwState = string.Empty;
try
{
if (losFormatter == null)
{
losFormatter = new LosFormatter();
}
string stateID = Page.Request["__VIEWSTATE_KEY"].ToString();
da = new DataAccess();
viewState = da.LoadVIEwState(stateID);
}
catch
{}
return losFormatter.Deserialize(vIEwState);
}
在ASP2.0中這行代碼基本是可以的,為什麼是基本呢,因為就是上面那行 ClientScript.RegisterHiddenField("__VIEWSTATE","");
有沒有這行,在ASP.Net1.1中都是可行的,我也是參考過別人的代碼,這行就這麼加入了,加了這行後,只是在頁面中多了個
<input type="hidden" name="__VIEWSTATE" value="" />
也就是運行後頁面的源文件中有兩個這樣的東西。去掉那行也可以,所以我不明白語句是用做什麼的,請明白的告訴我吧。但是在ASP.Net2.0中就不行,有如下錯誤:
The state information is invalid for this page and might be corrupted.
反正當時就是暈暈的,我以前從來沒有碰到過如是錯誤,去google也無所得,是啊,打死我也不知道是那句錯了啊,就這麼郁悶了兩天,問題無法解決,本人天生愚鈍的,我跟蹤視圖狀態的存入數據庫與從數據庫的讀取的整個過程,硬是找不到錯誤,我就反復思考這些代碼,惟有那行,我就是有點迷惑,為什麼還要頁面注冊一個“__VIEWSTATE”的隱藏域呢,於是我就注釋掉這行,居然可以運行了,所以我還是不明白那行是什麼用意。
當然我們還可以通過寫一個PageStatePersister新子類也可以完成上述功能,這是ASP.Net2.0新增的:
namespace PageAdapter
{
using System;
using System.IO;
using System.Security.Permissions;
using System.Web;
using System.Web.UI;
[AspNetHostingPermission(SecurityAction.Demand, Level = ASPNetHostingPermissionLevel.Minimal)]
public class DatabasePageStatePersister : PageStatePersister
{
public DatabasePageStatePersister(Page page): base(page)
{}
//
// Load VIEwState and ControlState.
//
public override void Load()
{
string vIEwState;
IStateFormatter formatter = this.StateFormatter;
DataAccess da = new DataAccess();
string stateID = base.Page.Request["__VIEWSTATE_KEY"].ToString();
viewState = da.LoadVIEwState(stateID);
Pair statePair = (Pair)formatter.Deserialize(vIEwState);
VIEwState = statePair.First;
ControlState = statePair.Second;
}
//
// Persist any VIEwState and ControlState.
//
public override void Save()
{
if (VIEwState != null || ControlState != null)
{
if (Page.Session != null)
{
string viewStateID = "VIEWSTATE#" + base.Page.Session.SessionID.ToString() + "#" + DateTime.Now.Ticks.ToString();
base.Page.ClientScript.RegisterHiddenField("__VIEWSTATE_KEY", vIEwStateID);
Pair statePair = new Pair(VIEwState, ControlState);
IStateFormatter formatter = this.StateFormatter;
// Serialize the statePair object to a string.
string serializedState = formatter.Serialize(statePair);
ViewStateData vsd = new VIEwStateData();
vsd.ViewStateID = vIEwStateID;
vsd.VIEwState = serializedState;
DataAccess da = new DataAccess();
string error = da.SaveVIEwState(vsd);
}
else
throw new InvalidOperationException("Session needed for StreamPageStatePersister.");
}
}
}
}
再有
重寫PageStatePersister屬性就可以了:
protected override PageStatePersister PageStatePersister
{
get
{
return new DatabasePageStatePersister(Page);
}
文件
這其實也跟數據庫的差不了多少,我這只講ASP.NET2.0的,在ASP.Net1.1也應該差不多,但我沒有寫代碼調試:
還是用那種寫PageStatePersister新子類的辦法:
namespace StreamPageAdapter
{
using System;
using System.IO;
using System.Security.Permissions;
using System.Web;
using System.Web.UI;
//
// The StreamPageStatePersister is an example vIEw state
// persistence mechanism that persists vIEw and control
// state on the Web server.
//
[AspNetHostingPermission(SecurityAction.Demand, Level = ASPNetHostingPermissionLevel.Minimal)]
public class StreamPageStatePersister : PageStatePersister
{
public StreamPageStatePersister(Page page): base(page)
{}
//
// Load VIEwState and ControlState.
//
public override void Load()
{
Stream stateStream = GetSecureStream();
// Read the state string, using the StateFormatter.
StreamReader reader = new StreamReader(stateStream);
IStateFormatter formatter = this.StateFormatter;
string fileContents = reader.ReadToEnd();
// Deserilize returns the Pair object that is serialized in
// the Save method.
Pair statePair = (Pair)formatter.Deserialize(fileContents);
VIEwState = statePair.First;
ControlState = statePair.Second;
reader.Close();
stateStream.Close();
}
//
// Persist any VIEwState and ControlState.
//
public override void Save()
{
if (VIEwState != null || ControlState != null)
{
if (Page.Session != null)
{
Stream stateStream = GetSecureStream();
StreamWriter writer = new StreamWriter(stateStream);
IStateFormatter formatter = this.StateFormatter;
Pair statePair = new Pair(VIEwState, ControlState);
// Serialize the statePair object to a string.
string serializedState = formatter.Serialize(statePair);
writer.Write(serializedState);
writer.Close();
stateStream.Close();
}
else
throw new InvalidOperationException("Session needed for StreamPageStatePersister.");
}
}
// Return a secure Stream for your environment.
private Stream GetSecureStream()
{
string path = @"d:\a.txt";
FileStream fs = new FileStream(path, FileMode.Open, FileAccess.ReadWrite);
return fs;
}
}
}
再重寫PageStatePersister屬性就可以了:
protected override PageStatePersister PageStatePersister
{
get
{
return new StreamPageStatePersister (Page);
}
通過上面的簡單介紹,我們應該有所了解了,只是要明白的是:在ASP.NET1.1中我們只能通過重寫age.SavePageStateToPersistenceMedium()和Page.LoadPageStateFromPersistenceMedium()來完成上述功能;而在ASP.Net2.0中,我們除了這外,還和通過寫PageStatePersister新子類和重寫PageStatePersister屬性來完成,我是沒有發現什麼不同,當然如果在下
面的內容你就明白,寫PageStatePersister新子類的真正用處了。
使用頁適配器
由於狀態持久性機制與自適應呈現和客戶端功能有關,因此提供MyPageAdapter激活 ASP.Net 應用程序的 DatabasePageStatePersister。最後,提供了一個浏覽器功能 (.browser) 文件來為特定類別的客戶端(在此例中為默認 Web 浏覽器)啟用 MyPageAdapter器。
這些內容請具體看我提供的源碼中的PageAdapter工程。看了就明白了。
using System.Security.Permissions;
using System.Web;
using System.Web.UI;
namespace PageAdapter
{
[AspNetHostingPermission(SecurityAction.Demand, Level = ASPNetHostingPermissionLevel.Minimal)]
public class MyPageAdapter : System.Web.UI.Adapters.PageAdapter
{
public override PageStatePersister GetStatePersister()
{
return new PageAdapter.DatabasePageStatePersister(Page);
}
}
}
最後,為了啟用 MyPageAdapter 適配器,您必須在 ASP.NET 應用程序的根目錄下創建一個名為 App_Browsers 的目錄,並在其中包括一個包含配置信息的 .browser 文件(其實這些都在你向工程中添加一個.browser文件時vs2005會自動給你完成的。配置文件中的 <refID元素指示該配置重寫為 Default.browser 配置文件中的默認浏覽器指定的值。此示例將 MyPageAdapter 用於 ASP.Net 網頁(但通常不使用適配器)。
<browsers>
<browser refID="Default" >
<controlAdapters>
<adapter
controlType="System.Web.UI.Page"
adapterType="PageAdapter.MyPageAdapter" />
</controlAdapters>
</browser>
</browsers>
這可以看源碼中的TestPageAdapter工程。該工程用來演示頁適配器的。
結束語
說得比較簡單,可能也不是很明白,至於各種持久機制的優劣,我也沒有專門測試過,而且最後一條“使用頁適配器”也不是屬於持久機制,只是用了也適配器,我們就不要重寫
PageStatePersister屬性了,我看來好像用處不是很大,因為我們可以把重寫PageStatePersister的動作放在頁面基類中,其它所有頁面都繼承這個基類就可以了,在我代碼中就是這麼做的,用這個頁適配器還麻煩了,當然我也並太清楚頁適配器的這個東東。
另外,對我的源碼做個簡單說明:
1. PageAdapter工程
DatabasePageStatePersister.cs:PageStatePersister類的子類
MyPageAdapter.cs:頁適配器
DataAccess.cs和VIEwSate.cs數據庫訪問的,屬於輔助類。
2. StreamPageAdapter工程
這個與上面的相似,不多說了
3. SaveStateToDatabase工程
StateInHiddenFIEld.ASPx:測試默認的存儲機制,就是在看頁面源文件時可以看到一大堆亂七八糟東西的。
StateInSession.ASPx:存儲機制為Session
StateInDatabase.ASPx:存儲機制數據庫,是重寫方法的那種,ASP.Net1.1,2.0都可以用的。
StateInDatabase2.ASPx:寫PageStatePersister新子類的並重寫PageStatePersister屬性的那種
StateInFile.ASPx:把VIEwState保存在服務器中某個文件夾中。
4. TestPageAdater工程。
用來測試也適配器用的。