Asp.Net中压缩ViewState的2种方法

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版的压缩/解压缩

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.IO;
using System.IO.Compression;

/// <summary>
///ViewStateCompression 的摘要说明
/// </summary>
public class ViewStateCompression
{
    public ViewStateCompression()
    {
        //
        //TODO: 在此处添加构造函数逻辑
        //
    }
    // 压缩
    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类:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

using System.IO;
using System.Web.UI;

/// <summary>
///BasePage 的摘要说明
/// </summary>
public class BasePage : System.Web.UI.Page
{
    public BasePage()
    {
        //
        //TODO: 在此处添加构造函数逻辑
        //
    }
    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() 方法。

改写上面的方法后的代码如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;

using System.IO;
using System.Web.UI;

/// <summary>
///BasePage 的摘要说明
/// </summary>
public class BasePage:System.Web.UI.Page
{
    public BasePage()
    {
        //
        //TODO: 在此处添加构造函数逻辑
        //
    }
    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,这在页面中很多数据源控件时,将提高了一定性能:

 

 

 

posted @ 2012-05-11 10:40  荡来荡去(allen)  阅读(447)  评论(0编辑  收藏  举报