如果你有在IE中查看當前瀏覽頁面HTML源代碼的習慣,你也許常會看到類似以下的代碼片斷:
<input type="hidden" name="__VIEWSTATE" value="dDwtMzU5NzUyMTQ1O3Q8O2w8aTwwPjs+O2w8dDw7bDxpPDA+Oz47bDx0PDtsPG
……
聰明的你一定會問,這是什么?有什么作用?它與本篇文章有何轉折親關系?各位看官,且聽我慢慢道來。
其實,這就是MS在asp.net應用ViewState技術的特征表現。為了頁面能在PostBack后依然能讀取服務器控件原有的狀態數據,Asp.net中使用了ViewState技術,而ViewState技術本質上是用一個默認名稱為"__VIEWSTATE的Hidden類型表單域來保存和傳遞數據(這些數據是經過了序列化后Base64編碼的字符串值,且是在方法Page.SavePageStateToPersistenceMedium輸出前保存、并由Page.LoadPageStateFromPersistenceMedium加載)。雖然我們可以通過三種級別來輕松禁用掉這些數據的往返傳遞:
Machine級 在machine.config中設置<pages enableViewStateMac='false' />
application級 在Web Applicatin的web.config中設置<pages enableViewStateMac='false' />
單頁面級 在該頁面中設置<%@Page enableViewStateMac='false' %>或通過代碼設置Page.EnableViewStateMac = false;
可是,如果我們完全能通過禁用ViewState來解決數據傳輸負擔而且不產生副作用的話,那MS的架構師們也不會傻到如此可愛的地步(可有可無的東東留它何用?),正因我們往往不能通過簡單的禁用來解決這個傳輸負擔問題,所以我們只能另辟路徑使之在網絡往返中傳輸量盡可能地小,于是,壓縮成了我們的首選。只要我們重載Page類的SavePageStateToPersistenceMedium()方法與LoadPageStateFromPersistenceMedium()方法,并在重載方法中對數據進行壓縮與解壓的處理即可。開源項目SharpZipLib提供的類GZipInputStream與GZipOutputStream進入了我們的視野,為了方便,不妨寫個類ComPRessionHelper,代碼如下:
1using System.IO;
2using ICSharpCode.SharpZipLib.GZip;
3
4namespace Ycweb.Components
5{
6 /**//// <summary>
7 /// Summary description for CompressionHelper.
8 /// </summary>
9 public class CompressionHelper
10 {
11 public CompressionHelper()
12 {
13 //
14 // TODO: Add constructor logic here
15 //
16 }
17
18 /**//// <summary>
19 /// 壓縮數據
20 /// </summary>
21 /// <param name="data">待壓縮的字節數組</param>
22 /// <returns>壓縮后的字節數組</returns>
23 public static byte[] CompressByte(byte[] data)
24 {
25 MemoryStream ms = new MemoryStream();
26 Stream s=new GZipOutputStream(ms);
27 s.Write( data, 0, data.Length );
28 s.Close();
29 return ms.ToArray();
30 }
31
32 /**//// <summary>
33 /// 解壓數據
34 /// </summary>
35 /// <param name="data">待解壓的字節數組</param>
36 /// <returns>解壓出的字節數組</returns>
37 public static byte[] DeCompressByte(byte[] data)
38 {
39 byte[] writeData = new byte[2048];
40 MemoryStream ms= new MemoryStream( data );
41 Stream sm = new GZipInputStream(ms) as Stream;
42 MemoryStream outStream = new MemoryStream();
43 while (true)
44 {
45 int size = sm.Read(writeData,0, writeData.Length );
46 if (size >0)
47 {
48 outStream.Write(writeData,0,size);
49 }
50 else
51 {
52 break;
53 }
54 }
55 sm.Close();
56 byte[] outArr = outStream.ToArray();
57 outStream.Close();
58 return outArr;
59 }
60 }
61} 然后我們在派生于類Page的頁面控制基類中重載方法LoadPageStateFromPersistenceMedium()與SavePageStateToPersistenceMedium(Object viewState),代碼如下:
1Load & Save ViewState Data#region Load & Save ViewState Data
2 protected override object LoadPageStateFromPersistenceMedium()
3 {
4//從自己注冊的隱藏域__SmartViewState中讀取數據
5 string viewState = Request.Form["__SmartViewState"];
6 byte[] bytes = Convert.FromBase64String(viewState);
7 //調用上面提供的靜態方法CompressionHelper.DeCompressByte()來解壓數據
8 bytes = CompressionHelper.DeCompressByte(bytes);
9 LosFormatter formatter = new LosFormatter();
10 return formatter.Deserialize(Convert.ToBase64String(bytes));
11
12 }
13
14 protected override void SavePageStateToPersistenceMedium(Object viewState)
15 {
16 LosFormatter formatter = new LosFormatter();
17 StringWriter writer = new StringWriter();
18 formatter.Serialize(writer, viewState);
19 string viewStateString = writer.ToString();
20 byte[] bytes = Convert.FromBase64String(viewStateString);
21 //調用上面提供的靜態方法CompressionHelper.CompressByte()來壓縮數據
22 bytes = CompressionHelper.CompressByte(bytes);
23
24 //注冊一個新的隱藏域__SmartViewState,你也可以自己定義
25 this.RegisterHiddenField("__SmartViewState", Convert.ToBase64String(bytes));
26 }
27#endregion
經過以上處理,web輸出頁面中的源代碼就是型如:
<input type="hidden" name="__SmartViewState" value="H4sIAPtPoNwA/81ce1PbWJb/ ……
<input type="hidden" name="__VIEWSTATE" value="" />
原來的隱藏域"__VIEWSTATE"值為空,而取而代之的是我們自己注冊的新的隱藏域"__SmartViewState"來存儲了經過壓縮后的字符串,這樣以來,提速效果是明顯的,結合我們的項目,象DG3G.COM的首頁原ViewState串值大約是28K,壓縮后為7K,而Acafa.com的首頁原ViewState串值大約是43K,壓縮后為8K。這樣的處理還是比較令人滿意的。當然,如果你覺得還不夠徹底,你還可以把ViewState串存儲在session中,不過這樣做,對服務器內存的壓力將陡增(尤其是訪問量較大的時候),建議還是不要輕易使用,如果你Web服務器內存有個10G、8G的,不妨試試。下面給出相關修改代碼:
將上述LoadPageStateFromPersistenceMedium()方法體中的代碼:
string viewState = Request.Form["__SmartViewState"];
修改為:
string viewState = Session["__SmartViewState"].ToString();
同時,將上述SavePageStateToPersistenceMedium()體中的代碼:
this.RegisterHiddenField("__SmartViewState", Convert.ToBase64String(bytes));
修改為:
Session["__SmartViewState"] = Convert.ToBase64String(bytes);
末了,以上代碼和方案均來自VS2003開發實踐,對VS2005是否合適,本人不做任何承諾,不過如果你能從以上方案中有所收獲,我將感到無比的高興。
新聞熱點
疑難解答