当前位置:首页 > 网络编程 > WEB编程 > ASP.net >  小议优化ASP.NET应用性能之ViewState篇

 小议优化ASP.NET应用性能之ViewState篇

点击次数:22 次 发布日期:2008-11-26 12:53:54 作者:源代码网
源代码网推荐     
源代码网推荐   如果你有在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是否合适,本人不做任何承诺,不过如果你能从以上方案中有所收获,我将感到无比的高兴。
源代码网推荐  
源代码网推荐  
源代码网推荐  
源代码网推荐    做人要厚道,请注明转自酷网动力(www.ASPCOOL.COM)。
源代码网推荐


源代码网供稿.
网友评论 (0)
会员中心
网络编程
本站推荐
网络编程之精华