• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

ASP.NET缓存:方法和最好实践(转载)

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

【简 介】
  在 asp.net 提供的许多特性中,缓存支持无疑是我最欣赏的特性,我这样说当然是有充分理由的。相比 asp.net 的任何其他特性,缓存对应用程式的性能具备最大的潜在影响,利用缓存和其他机制,asp.net 研发人员能够接受使用开销很大的控件(例如,datagrid)构建站点时的额外开销,而不必担心性能会受到太大的影响。为了在应用程式中最大程度地利用缓存,您应该考虑在任何程式级别上都实现缓存的方法。


  实现  

  要实现页面输出缓存,只要将一条 outputcache 指令添加到页面即可。  

  <%@ outputcache duration="60" varybyparam="*" %>  

  如同其他页面指令相同,该指令应该出现在 aspx 页面的顶部,即在任何输出之前。他支持五个属性(或参数),其中两个是必需的。 

  duration

  必需属性。页面应该被缓存的时间,以秒为单位。必须是正整数。 

  location

  指定应该对输出进行缓存的位置。假如要指定该参数,则必须是下列选项之一:any、client、downstream、none、server 或 serverandclient。 

  varybyparam

   必需属性。request 中变量的名称,这些变量名应该产生单独的缓存条目。"none" 表示没有变动。"*" 可用于为每个不同的变量数组创建新的缓存条目。变量之间用 ";" 进行分隔。  

  varybyheader

   基于指定的标头中的变动改变缓存条目。  

  varybycustom

   允许在 global.asax 中指定自定义变动(例如,"browser")。  

  利用必需的 duration 和 varybyparam 选项的组合能够处理大多数情况。例如,假如您的产品目录允许用户基于 categoryid 和页变量查看目录页,您能够用参数值为 "categoryid;page" 的 varybyparam 将产品目录缓存一段时间(假如产品不是随时都在改变,一小时还是能够接受的,因此,持续时间是 3600 秒)。这将为每个种类的每个目录页创建单独的缓存条目。每个条目从其第一个请求算起将维持一个小时。 

  varybyheader 和 varybycustom 主要用于根据访问页面的客户端对页面的外观或内容进行自定义。同一个 url 可能需要同时为浏览器和移动电话客户端呈现输出,因此,需要针对不同的客户端缓存不同的内容版本。或,页面有可能已针对 ie 进行了优化,但需要能针对 netscape 或 opera 完全降低优化(而不但仅是破坏页面)。后一个例子很普遍,我们将提供一个说明如何实现此目标的示例: 

  示例:varybycustom 用于支持浏览器自定义  

  为了使每个浏览器都具备单独的缓存条目,varybycustom 的值能够配置为 "browser"。此功能已内置在缓存模块中,并且将针对每个浏览器名称和主要版本插入单独的页面缓存版本。 

  <%@ outputcache duration="60" varybyparam="none" varybycustom="browser" %> 

  片段缓存,用户控件输出缓存

  缓存整个页面通常并不可行,因为页面的某些部分是针对用户定制的。但是,页面的其他部分是整个应用程式共有的。这些部分最适合使用片段缓存和用户控件进行缓存。菜单和其他布局元素,尤其是那些从数据源动态生成的元素,也应该用这种方法进行缓存。假如需要,能够将缓存的控件配置为基于对其控件(或其他属性)的更改或由页面级输出缓存支持的任何其他变动进行改变。使用同一组控件的几百个页面还能够共享那些控件的缓存条目,而不是为每个页面保留单独的缓存版本。 



  实现  

  片段缓存使用的语法和页面级输出缓存相同,但其应用于用户控件(.ascx 文档)而不是 web 窗体(.aspx 文档)。除了 location 属性,对于 outputcache 在 web 窗体上支持的任何属性,用户控件也同样支持。用户控件还支持名为 varybycontrol 的 outputcache 属性,该属性将根据用户控件(通常是页面上的控件,例如,dropdownlist)的成员的值改变该控件的缓存。假如指定了 varybycontrol,能够省略 varybyparam。最后,在默认情况下,对每个页面上的每个用户控件都单独进行缓存。但是,假如一个用户控件不随应用程式中的页面改变,并且在任何页面都使用相同的名称,则能够应用 shared="true" 参数,该参数将使用户控件的缓存版本供任何引用该控件的页面使用。

  示例  

  <%@ outputcache duration="60" varybyparam="*" %>  

  该示例将缓存用户控件 60 秒,并且将针对查询字符串的每个变动、针对此控件所在的每个页面创建单独的缓存条目。  

  <%@ outputcache duration="60" varybyparam="none"

  varybycontrol="categorydropdownlist" %>  

  该示例将缓存用户控件 60 秒,并且将针对 categorydropdownlist 控件的每个不同的值、针对此控件所在的每个页面创建单独的缓存条目。  

  <%@ outputcache duration="60" varybyparam="none" varybycustom="browser"

  shared="true %>  

  最后,该示例将缓存用户控件 60 秒,并且将针对每个浏览器名称和主要版本创建一个缓存条目。然后,每个浏览器的缓存条目将由引用此用户控件的任何页面共享(只要任何页面都用相同的 id 引用该控件即可)。  
  页面级和用户控件级输出缓存的确是一种能够迅速而简便地提高站点性能的方法,但是在 asp.net 中,缓存的真正灵活性和强大功能是通过 cache 对象提供的。使用 cache 对象,您能够存储任何可序列化的数据对象,基于一个或多个依赖项的组合来控制缓存条目到期的方式。这些依赖项能够包括自从项被缓存后经过的时间、自从项上次被访问后经过的时间、对文档和/或文档夹的更改连同对其他缓存项的更改,在略作处理后还能够包括对数据库中特定表的更改。 

  在 cache 中存储数据  

  在 cache 中存储数据的最简单的方法就是使用一个键为其赋值,就像 hashtable 或 dictionary 对象相同:  

  cache["key"] = "value";  

  这种做法将在缓存中存储项,同时不带任何依赖项,因此他不会到期,除非缓存引擎为了给其他缓存数据提供空间而将其删除。要包括特定的缓存依赖项,可使用 add() 或 insert() 方法。其中每个方法都有几个重载。add() 和 insert() 之间的唯一区别是,add() 返回对已缓存对象的引用,而 insert() 没有返回值(在 c# 中为空,在 vb 中为 sub)。  

  示例 

  cache.insert("key", myxmlfiledata, new

  system.web.caching.cachedependency(server.mappath("users.xml")));  

  该示例可将文档中的 xml 数据插入缓存,无需在以后请求时从文档读取。 cachedependency 的作用是确保缓存在文档更改后立即到期,以便能够从文档中提取最新数据,重新进行缓存。假如缓存的数据来自若干个文档,还能够指定一个文档名的数组。  

  cache.insert("dependentkey", mydependentdata, new

  system.web.caching.cachedependency(new string[] {}, new string[]

  {"key"}));  

  该示例可插入键值为 "key" 的第二个数据块(取决于是否存在第一个数据块)。假如缓存中不存在名为 "key" 的键,或假如和该键相关联的项已到期或被更新,则 "dependentkey" 的缓存条目将到期。  

  cache.insert("key", mytimesensitivedata, null,

  datetime.now.addminutes(1), timespan.zero);  

  绝对到期:此示例将对受时间影响的数据缓存一分钟,一分钟过后,缓存将到期。注意,绝对到期和滑动到期(见下文)不能一起使用。  

  cache.insert("key", myfrequentlyaccesseddata, null,

  system.web.caching.cache.noabsoluteexpiration,

  timespan.fromminutes(1));  

  滑动到期:此示例将缓存一些频繁使用的数据。数据将在缓存中一直保留下去,除非数据未被引用的时间达到了一分钟。注意,滑动到期和绝对到期不能一起使用。 



  更多选项  

  除了上面提到的依赖项,我们还能够指定项的优先级(依次为 low、high、notremovable,他们是在 system.web.caching.cacheitempriority 枚举中定义的)连同当缓存中的项到期时调用的 cacheitemremovedcallback 函数。大多数时候,默认的优先级已足够了 — 缓存引擎能够正常完成任务并处理缓存的内存管理。cacheitemremovedcallback 选项考虑到一些很有趣的可能性,但实际上他很少使用。但是,为了说明该方法,我将提供他的一个使用示例:  

  cacheitemremovedcallback 示例
 
  system.web.caching.cacheitemremovedcallback callback = new system.web.caching.cacheitemremovedcallback (onremove);

  cache.insert("key",myfile,null,

  system.web.caching.cache.noabsoluteexpiration,

  timespan.zero,

  system.web.caching.cacheitempriority.default, callback);

  . . .

  public static void onremove(string key,

  object cacheitem,

  system.web.caching.cacheitemremovedreason reason)

  {

  appendlog("the cached value with key " + key +

  " was removed from the cache. reason: " +

  reason.tostring());

  }  

  该示例将使用 appendlog() 方法(这里不讨论该方法,请参阅 writing entries to event logs)中定义的任何逻辑来记录缓存中的数据到期的原因。通过在从缓存中删除项时记录这些项并记录删除的原因,您能够确定是否在有效地使用缓存或您是否可能需要增加服务器上的内存。注意,callback 是个静态(在 vb 中为 shared)方法,建议使用该方法的原因是,假如不使用他,保存回调函数的类的实例将保留在内存中,以支持回调(对 static/shared 方法则没有必要)。  
  该特性有一个潜在的用处 — 在后台刷新缓存的数据,这样用户永远都不必等待数据被填充,但数据始终保持相对较新的状态。但实际上,此特性并不适用于当前版本的缓存 api,因为在从缓存中删除缓存的项之前,不触发或不完成回调。因此,用户将频繁地发出尝试访问缓存值的请求,然后发现缓存值为空,不得不等待缓存值的重新填充。我希望在未来的 asp.net 版本中看到一个附加的回调,能够称为 cacheditemexpiredbutnotremovedcallback,假如定义了该回调,则必须在删除缓存项之前完成执行。  

  缓存数据引用模式  

  每当我们尝试访问缓存中的数据时,都应该考虑到一种情况,那就是数据可能已不在缓存中了。因此,下面的模式应该普遍适用于您对缓存的数据的访问。在这种情况下,我们假定已缓存的数据是个数据表。  

  public datatable getcustomers(bool bypasscache)

  {

  string cachekey = "customersdatatable";

  object cacheitem = cache[cachekey] as datatable;

  if((bypasscache)    (cacheitem == null))

  {

  cacheitem = getcustomersfromdatasource();

  cache.insert(cachekey, cacheitem, null,

  datetime.now.addseconds(getcachesecondsfromconfig(cachekey),

  timespan.zero);

  }

  return (datatable)cacheitem;

  }  



  关于此模式,有以下几点需要注意:   

  ? 某些值(例如,cachekey、cacheitem 和缓存持续时间)是一次定义的,并且只定义一次。 

  ? 能够根据需要跳过缓存 — 例如,当注册一个新客户并重定向到客户列表后,最好的做法可能就是跳过缓存,用最新数据重新填充缓存,该数据包括新插入的客户。   

  ? 缓存只能访问一次。这种做法能够提高性能,并确保不会发生 nullreferenceexceptions,因为该项在第一次被检查时是存在的,但第二次检查之前就已到期了。   

  ? 该模式使用强类型检查。c# 中的 "as" 运算符尝试将对象转换为类型,假如失败或该对象为空,则只返回 null(空)。   

  ? 持续时间存储在配置文档中。在理想的情况下,任何的缓存依赖项(无论是基于文档的,或是基于时间的,还是其他类型的依赖项)都应该存储在配置文档中,这样就能够进行更改并轻松地测量性能。我还建议您指定默认缓存持续时间,而且,假如没有为所使用的 cachekey 指定持续时间,就让 getcachesecondsfromconfig() 方法使用该默认持续时间。   

  相关的代码示例是个 helper 类,他将处理上述任何情况,但允许通过一行或两行代码访问缓存的数据。请下载 cachedemos.msi。  

  小结

  缓存能够使应用程式的性能得到很大的提高,因此在设计应用程式连同对应用程式进行性能测试时应该予以考虑。应用程式总会或多或少地受益于缓存,当然有些应用程式比其他应用程式更适合使用缓存。对 asp.net 提供的缓存选项的深刻理解是任何 asp.net 研发人员应该掌控的重要技巧。 



  尽早缓存;经常缓存  

  您应该在应用程式的每一层都实现缓存。向数据层、业务逻辑层、ui 或输出层添加缓存支持。内存现在很便宜 — 因此,通过以智能的方式在整个应用程式中实现缓存,能够获得很大的性能提高。

  缓存能够掩盖许多过失  

  缓存是一种无需大量时间和分析就能够获得"足够良好的"性能的方法。这里再次强调,内存现在很便宜,因此,假如您能通过将输出缓存 30 秒,而不是花上一整天甚至一周的时间尝试优化代码或数据库就能够获得所需的性能,您肯定会选择缓存解决方案(假设能够接受 30 秒的旧数据)。缓存正是那些利用 20% 付出获得 80% 回报的特性之一,因此,要提高性能,应该首先想到缓存。但是,假如设计很糟糕,最终却有可能带来不良的后果,因此,您当然也应该尽量正确地设计应用程式。但假如您只是需要立即获得足够高的性能,缓存就是您的最好选择,您能够在以后有时间的时候再尽快重新设计应用程式。  

  页面级输出缓存

  作为最简单的缓存形式,输出缓存只是在内存中保留为响应请求而发送的 html 的副本。其后再有请求时将提供缓存的输出,直到缓存到期,这样,性能有可能得到很大的提高(取决于需要多少开销来创建原始页面输出 - 发送缓存的输出总是很快,并且比较稳定)。  


以上内容由 华夏名网 收集整理,如转载请注明原文出处,并保留这一部分内容。

http://www.sudu.cn/info/html/edu/net/20071226/26662.html


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
Asp.NetMVCsession跨域发布时间:2022-07-12
下一篇:
动态加载Asp.net分页控件发布时间:2022-07-12
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap