ASP.NET缓存策略经验谈
要提升ASP.NET应用程序的性能,最简单、最有效的方式就是使用内建的缓存引擎。虽然也能构建自己的缓存,但由于缓存引擎已提供了如此多的功能,所以完全不必如此麻烦。在很大程度上,ASP.NET开发者在Web应用程序中,能将缓存引擎的功能直接包装到自己的数据表示及访问类中。如本文所述,整个过程其实非常简单。
ASP.NET的缓存引擎支持三种类型的缓存:
整页输出缓存是在一个页被首次请求时,将整个页呈现好的HTML内容缓存下来。后续请求将直接取用缓存拷贝。
部分缓存是指缓存一部分HTML内容,这类似一个Web用户控件的输出。之所以叫这样的一个名字,是因为我们一般说“将一部分HTML提交给一个页”。
数据缓存关注的是单独的变量或数据项的缓存。它在比以上两种缓存类型都要低的一个级别上工作。
整页输出缓存
整页输出缓存是最简单的缓存类型,它只要求为准备缓存的页添加一个预处理指令OutputCache。使用这种缓存,就不必重新处理一个页的Init,Load,PreRender,Render以及Unload事件。假如那些事件要访问像数据库那样的一个后端系统,那么节省的时间将是非常可观的。ASP.NET可缓存一个页的几个变体,并将每个页都与后续请求相关联。所有这些都由OutputCache预处理指令来控制,该命令要放在一个ASPX页的顶部,并采用以下格式:
<%@ OutputCache
Duration="#ofseconds"
Location="Any | Client | Downstream | Server | None"
VaryByCustom="browser | customstring"
VaryByHeader="headers"
VaryByParam="parameter name" %>
OutputCache预处理指令最多可利用5个相关的属性来控制缓存行为。表A总结了这些属性及其含义。
表A属性说明
Duration 该属性指定了网页要缓存的时间(以秒为单位)。通过为一个页设置该属性,可为来自对象HTTP响应建立一个过期策略。这是一个必需属性。如果不包括它,会发生解析器错误。注意它指定的是绝对过期时间,而非周期性过期。
Location 用这个属性指示Web服务器和下游设备(比如代理服务器和浏览器)来缓存页面内容。默认为Any,该属性并非必需的。
VaryByParam 该属性使用由分号分隔的字符串列表来改变输出缓存。默认情况下,这些字符串对应于随GET方法属性发送的查询字符串值,或对应于使用POST方法发送的一个参数。如果将该属性设为多个参数,输出缓存将针对指定的每个参数,包含所请求的文档的一个不同版本。可能的值包括None、*以及任何有效的查询字符串或POST参数名。如果没有设置其他属性,该属性就是必需的。
VaryByCustom 对自定义输出缓存需求进行表示的任何文本都要包含在这个属性中。如果为属性指派浏览器的一个值,缓存会根据浏览器名称及主版本信息而发生变化。如果输入一个自定义字符串,就必须在应用程序的Global.asax文件中覆盖GetVaryByCustomString方法。
VaryByHeader 该属性使用由分号分隔的HTTP标头列表来改变输出缓存。如果属性设为多个标头,输出缓存将针对指定的每个标头,包含所请求文档的一个不同的版本。
OutputCache属性
OutputCache预处理指令可在内存中方便地创建静态页的单一版本,例如:
<%@ OutputCache Duration=“300" VaryByParam="None" %>
还可根据id查询字符串参数来缓存多个版本,例如:
<%@ OutputCache Duration=“300" VaryByParam="id" %>
在上述两个例子中,网页会在5分钟(300秒)后从缓存中清除,该数字是由Duration属性指定的。
根据自定义信息来缓存
使用VaryByCustom属性,可根据你提供的自定义信息来创建一个页的缓存版本。例如,要为每种类型的浏览器创建网页的一个不同版本,可使用以下预处理指令:
<%@ OutputCache Duration=“300" VaryByParam="None" VaryByCustom=“browser”%>
使用VaryByCustom,还可根据应用程序特有的信息来缓存网页。例如,要根据当前用户的部门名称来创建网页的缓存版本,可使用以下预处理指令:
<%@ OutputCache Duration=“300" VaryByParam="None" VaryByCustom=“department”%>
然后,在用于Global.asax文件的代码隐藏类中,可覆盖GetVaryByCustomString方法,如以下VB.NET代码所示:
Public Overrides Function GetVaryByCustomString( _
ByVal context As System.Web.HttpContext, _
ByVal custom As String) As String
If custom = "department" Then
' return the department for the current user
End If
End Function
这里向GetVaryByCustomString方法传递了在预处理指令中使用的自定义值。ASP.NET运行库调用该方法时,你可使用自己的算法,根据输入值来判断当前用户所在的部门,并从方法中返回它。
根据标头信息来缓存
使用VaryByHeader属性,可根据网页接收到的一个或多个HTTP标头来创建一个网页的多个缓存版本。例如,要想根据浏览器的语言来创建和缓存不同的版本,可使用以下预处理指令:
<%@ OutputCache Duration="60" VaryByParam="None" VaryByHeader="Accept-Language" %>
检查标头
要查看随同特定Web请求发送的各个HTTP标头的值,可打开ASP.NET页的跟踪功能,具体做法是在网页的Page预处理指令中设置trace和traceMode属性。另外,也可通过Visual Studio .NET的属性页来设置。
除了声明性地缓存整个网页,还可使用由Page类揭示的Response对象的Cache属性来程序化地设置一个页的缓存选项。Cache属性揭示了用于网页的HttpCachePolicy对象,可用它在服务器上将缓存时间设为5分钟,如以下C# 代码段所示:
private void Page_Init( object sender, System.EventArgs e )
{
Response.Cache.SetCacheability(HttpCacheability.Server);
Response.Cache.SetExpires(DateTime.Now.AddSeconds(300));
Response.Cache.SetValidUntilExpires(True)
}
部分缓存
部分缓存允许只缓存部分HTML,它的效率很高,因为一个页中经常变化的部分可与那些不经常变化部分的部分合并到一起,同时仍能从缓存中检索静态的部分。一个实例是将“部分缓存”应用于需要调用XML Web服务的Web用户控件。这样可获得很高的效率,因为它避免了你的网站与Web服务过度紧密地耦合,同时还能显著提升性能。
要使用部分缓存,同样可在HTML页的顶部放置一条OutputCache预处理指令。但这一次,我们准备把它放到Web用户控件的ASCX页中。注意使用部分缓存时,Location和VaryByHeader属性将不再支持,但新增了对VaryByControl属性的支持。
使用VaryByControl属性,可用一个由分号分隔的列表来指定用户控件的一个或多个属性。可为属性值的每一种组合来创建缓存的版本。例如,假定你的用户控件揭示出一个自定义的State属性,它控制着要显示用户控件的哪些元素。使用以下预处理指令,可针对State的每个值来缓存控件所具体呈现的一个版本:
<%@ OutputCache Duration="300" VaryByControl="State" %>
但在缓存Web用户控件时,记住ASP.NET运行库会直接用缓存的HTML来替换实际控件,忽略平常会发生的任何控件处理。这暗示着在网页中执行的代码不能程序化地操纵一个缓存的用户控件或者它的任何属性。换言之,Web用户控件必须能完全自主,并能通过它的Load和Init事件来初始化自己,以便有效地捍妗?/P>
也可声明性地完成部分缓存,方法是使用一个属性而不是使用OutputCache预处理指令。在代码隐藏文件中,PartialCaching属性可放在从UserControl派生的一个类中,使ASP.NET运行库能够读取它,并相应地缓存呈现好的HTML。例如,以下来自代码隐藏类的声明能根据查询字符串中的id值,将Web用户控件缓存5分钟。
<PartialCaching(300, "id", Nothing, Nothing)> _
Public MustInherit Class MyHeaderControl
Inherits System.Web.UI.UserControl
数据缓存
ASP.NET缓存引擎支持的最后一种缓存类型是“数据缓存”。根据定义,它的工作级别要低于整页输出缓存和部分缓存。假如几个网页都要使用相同的数据(例如一个产品列表),但要以不同方式来显示这些数据,就可考虑使用这种缓存。当然,数据缓存之所以具有性能优势,是因为减少了对后端数据库的调用次数。
为了在缓存中添加一个项,需使用Page或UserControl类的Cache属性,因为这两个类最终都是从Control类派生的。Cache属性揭示出了System.Web.Caching.Cache对象,利用它可将数据当作键和值的一个组合来存储。使用该属性,开发者可编写代码来填充一个项,并把它放到缓存中;如果该项已经存在,就直接将其从缓存中取出。如以下C#代码所示。
DataTable dt = null;
if (this.Cache["Products"] == null)
{
// Go get the data from the database
this.Cache.Insert("Products", dt, null, DateTime.Now.AddHours(6), TimeSpan.Zero);
}
else
{
dt = this.Cache["Products"] As DataTable;
}
上例首先检查具有Products键的项是否在缓存中。如果不在,就从后端数据库检索一个ADO.NET DataTable,并使用Insert方法把它放到缓存中。本例使用的是Insert的一个重载版本,它允许为缓存对象指定一个绝对过期时间(6小时),而不是指定一个周期性的过期。相反,如果缓存中已经有这个项,就将其取回,并使用As表达式,将其强制转换回一个DataTable。
对ADO.NET检索到的数据进行缓存时,注意既可像上例那样缓存DataTable对象,也可缓存整个DataSet对象,因为两种对象都同任何数据源完全地断开,不会保持数据库连接。对数据读取器(比如SqlDataReader)进行缓存似乎更好一些,因为它们只使用一次(它们是“只进”的读取器),而且在打开的情况下将一直占据一个数据库连接。
ASP.NET缓存引擎强大的灵活性和功能使其成为创建高性能ASP.NET应用程序时最重要的特性之一。根据本文提供的基本信息,你可在自己的应用程序轻松引入缓存引擎功能。
MSDN对此问题的讲解:ASP.NET 缓存:方法和最佳实践