轉帖|其它|編輯:郝浩|2011-06-27 14:48:06.000|閱讀 519 次
概述:asp.net ViewState 是一種新的狀態服務,可供開發人員基于每個用戶來跟蹤 UI 狀態,該輔助數據被存儲在一個名為 __VIEWSTATE 的隱藏字段中。
# 界面/圖表報表/文檔/IDE等千款熱門軟控件火熱銷售中 >>
asp.net ViewState 是一種新的狀態服務,可供開發人員基于每個用戶來跟蹤 UI 狀態,該輔助數據被存儲在一個名為 __VIEWSTATE 的隱藏字段中。
當然, ViewState 在 ASP.NET 中有個重要的角色。如果使用恰當,它能夠簡化頁面開發,改進用戶與站點的交互。如果置之不理,它能夠顯著增加站點響應大小,在連接速度慢的情況下,使您的響應時間更加緩慢。因為瀏覽器的每次回發都會導致ViewState 逐漸增加您的頁面大小,從而導致性能問題。因此,ASP.NET 2.0 的發布帶來了 ViewState 機制的一些改進,這使得 ViewState 使用更簡單,又不會防礙站點性能。這些改進包括:減少編碼數量,采用控件狀態從內容中分離出行為狀態,以及智能集成數據綁定控件。你可以在不需要維護控件狀態的情況下通過禁用的控件(EnableViewState = false )解決這個問題。 然而,很多情況下保持控件的狀態是必需的,壓縮的ViewState有助于提高性能。
方法一:使用System.IO.ComPRession
System.IO.Compression 命名空間包含提供基本的流壓縮和解壓縮服務的類。
此命名空間包含2個類分別為:
DeflateStream 提供用于使用 Deflate 算法壓縮和解壓縮流的方法和屬性。
GZipStream 提供用于壓縮和解壓縮流的方法和屬性。
在下面的演示代碼中,我們創建一個ViewStateCompression類,包含2個方法,并都返回byte[]數據:
1.GZipStream版的壓縮/解壓縮
namespace ASPNET_ViewState.Code {
public class ViewStateCompression {
public ViewStateCompression() {
//
// TODO: Add constructor logic here
//
}
// 壓縮
public static byte[] Compress( byte[] data ) {
MemoryStream output = new MemoryStream();
GZipStream gzip = new GZipStream( output,
CompressionMode.Compress, true );
gzip.Write( data, 0, data.Length );
gzip.Close();
return output.ToArray();
}
// 解壓縮
public static byte[] Decompress( byte[] data ) {
MemoryStream input = new MemoryStream();
input.Write( data, 0, data.Length );
input.Position = 0;
GZipStream gzip = new GZipStream( input,
CompressionMode.Decompress, true );
MemoryStream output = new MemoryStream();
byte[] buff = new byte[ 64 ];
int read = -1;
read = gzip.Read( buff, 0, buff.Length );
while ( read > 0 ) {
output.Write( buff, 0, read );
read = gzip.Read( buff, 0, buff.Length );
}
gzip.Close();
return output.ToArray();
}
}
}
2.執行ViewStateCompression類
想使用ViewStateCompression的壓縮和解壓頁面ViewState的功能,我們必須重寫 System.Web.UI.Page 的 SavePageStateToPersistenceMedium() 和LoadPageStateFromPersistenceMedium() 方法。
SavePageStateToPersistenceMedium 方法 可以反序列化的ViewState,它接受一個 ViewState對象的參數。
LoadPageStateFromPersistenceMedium 方法 可以序列化ViewState,它接受一個Base64編碼的字符串參數。
重寫代碼如下,新建了一個繼承與System.Web.UI.Page的BasePage類:
namespace ASPNET_ViewState.Code {
public class BasePage : System.Web.UI.Page {
public BasePage() {
}
protected override void SavePageStateToPersistenceMedium( object pageViewState ) {
LosFormatter losformatter = new LosFormatter();
StringWriter sw = new StringWriter();
losformatter.Serialize( sw, pageViewState );
string viewStateString = sw.ToString();
byte[] b = Convert.FromBase64String( viewStateString );
b = ViewStateCompression.Compress( b );
// ----ClientScript.RegisterHiddenField( "__ZIPSTATE", Convert.ToBase64String( b ) );
//
// 兼容ASP.NET Ajax 的ViewState壓縮
ScriptManager.RegisterHiddenField( this, "__ZIPSTATE", Convert.ToBase64String( b ) );
}
// 序列化ViewState
protected override object LoadPageStateFromPersistenceMedium() {
string custState = Request.Form[ "__ZIPSTATE" ];
byte[] b = Convert.FromBase64String( custState );
b = ViewStateCompression.Decompress( b );
LosFormatter losformatter = new LosFormatter();
return losformatter.Deserialize( Convert.ToBase64String( b ) );
}
}
}
經過上述的方法后,你的ViewState可能會減少30-40%.
3.ViewState SEO
ViewState會影響SEO,當然,影響并不是很大。搜索引擎在搜錄頁面的時候,從頁面源文件第一個字符開始,到100K的位置,后面的收錄不是很友好,甚至還會出現收錄問題。因此,在前100K的時候,我們可以考慮將ViewState 移到頁面的底部, 之前。參考代碼請下載本文的示例,這里給出效果圖參考:
移動之前:
移動后:
方法二:使用session完全刪除ViewState
使用這個方法即可以完全刪除ViewState,又可以完全保存ViewState的優勢,還可以減少客戶端要求下載的多余字節,還可以搜索引擎解決收錄問題。原理是通過利用Session允許在服務器上保存ViewState的方法重寫上面的SavePageStateToPersistenceMedium() 和LoadPageStateFromPersistenceMedium() 方法。
改寫上面的方法后的代碼如下:
namespace ASPNET_ViewState.Code {
public class BasePage : System.Web.UI.Page {
public BasePage() {
}
protected override void SavePageStateToPersistenceMedium( object pageViewState ) {
MemoryStream ms = new MemoryStream();
LosFormatter m_formatter = new LosFormatter();
m_formatter.Serialize( ms, pageViewState );
ms.Position = 0;
StreamReader sr = new StreamReader( ms );
string viewStateString = sr.ReadToEnd();
byte[] ViewStateBytes = Convert.FromBase64String( viewStateString );
ViewStateBytes = ViewStateCompression.Compress( ViewStateBytes );
Session[ "ViewState" ] = Convert.ToBase64String( ViewStateBytes );
ms.Close();
return;
}
// 序列化ViewState
protected override object LoadPageStateFromPersistenceMedium() {
object viewStateBag;
string m_viewState = ( string )Session[ "ViewState" ];
byte[] ViewStateBytes = Convert.FromBase64String( m_viewState );
ViewStateBytes = ViewStateCompression.Decompress( ViewStateBytes );
LosFormatter m_formatter = new LosFormatter();
try {
viewStateBag = m_formatter.Deserialize( Convert.ToBase64String( ViewStateBytes ) );
}
catch ( Exception ex ) {
//Log.Insert( "頁面Viewtate是空。" );
viewStateBag = string.Empty;
}
return viewStateBag;
}
}
}
改寫后我們可以從下圖看到ViewState完全被刪除了,而且傳輸到客戶端的數據從1006B 減少 到750B,這在頁面中很多數據源控件時,將提高了一定性能:
本站文章除注明轉載外,均為本站原創或翻譯。歡迎任何形式的轉載,但請務必注明出處、不得修改原文相關鏈接,如果存在內容上的異議請郵件反饋至chenjj@fc6vip.cn
文章轉載自:網絡轉載