您的位置:首页 > 编程语言 > ASP

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

2012-05-11 10:40 267 查看
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,这在页面中很多数据源控件时,将提高了一定性能:





内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: