當然, ViewState 在 ASP.NET 中有個重要的角色。如果使用恰當,它能夠簡化頁面開發,改進用戶與站點的交互。如果置之不理,它能夠顯著增加站點響應大小,在連接速度慢的情況下,使您的響應時間更加緩慢。因為瀏覽器的每次回發都會導致ViewState 逐漸增加您的頁面大小,從而導致性能問題。因此,ASP.NET 2.0 的發布帶來了 ViewState 機制的一些改進,這使得 ViewState 使用更簡單,又不會防礙站點性能。這些改進包括:減少編碼數量,采用控件狀態從內容中分離出行為狀態,以及智能集成數據綁定控件。你可以在不需要維護控件狀態的情況下通過禁用的控件(EnableViewState = false )解決這個問題。 然而,很多情況下保持控件的狀態是必需的,壓縮的ViewState有助于提高性能。
ASPNET_ViewState.rar 34KB.
方法一:使用System.IO.ComPRession
System.IO.Compression 命名空間包含提供基本的流壓縮和解壓縮服務的類。
此命名空間包含2個類分別為:
在下面的演示代碼中,我們創建一個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(); } }}
1B.DeflateStream版的壓縮/解壓縮:請下載本文的示例代碼。
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,這在頁面中很多數據源控件時,將提高了一定性能:
![]()
ASPNET_ViewState.rar 34KB.
新聞熱點
疑難解答