如果你有在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是否合適,本人不做任何承諾,不過如果你能從以上方案中有所收獲,我將感到無比的高興。
新聞熱點
疑難解答
圖片精選