在ASP.NET中构建和使用自定义的
OutputCache提供程序
Brandon Satrom
下载代码示例
如果您是一位Web开发人员您过去可能使用过ASP.NET提供的输出缓存功能。 ASP.NET输出缓存功能是随着Microsoft .NET Framework的第一个版本推出的该功能通过从缓存中检索向站点访问者提供的内容以及避免重新执行页面或控制器提高了向站点访问者提供内容方面的性能。 当返回您不经常更新的数据或者返回一段时间后将过期的数据时该功能不需要您的应用程序执行耗费大量资源的数据库调用操作。
ASP.NET输出缓存使用的是内存存储机制并且在NET Framework 4出现之前您无法使用您自己的实现覆盖或替代默认缓存。 现在借助新的OutputCacheProvider类型您可以在ASP.NET中实现您自己的缓存页面输出机制。
在本文中我将为您介绍两种自定义机制。 首先我将使用MongoDB 一种常用的面向文档数据库在一个简单的ASP.NET MVC应用程序中创建我自己的提供程序以方便输出缓存。然后我会使用同一个应用程序快速交换我的自定义提供程序 以便利用Windows AzureAppFabric的功能—具体地说就是在云中利用Windows Azure基础结构提供分布式内存缓存的新DistributedCache提供程序。
ASP.NET中的输出缓存
在ASP.NET Web窗体应用程序中可以通过向任意ASP.NET页面或用户控件添加
OutputCache Page指令来配置输出缓存
1. <%@ OutputCache Duration="60" Location="Any" VaryByParam="name"
%>
2.
对于ASP.NET MVC应用程序输出缓存是使用ASP.NET MVC附带的操作筛选器来提供的该操作筛选器可用作任何控制器操作的一个属性
1. [OutputCache(Duration=60, VaryByParam="none") ]
2.
“Duration”和“VaryByParam”在ASP.NET MVC 1和2应用程序中是必需的VaryByParam在ASP.NET MVC 3中是可选的 这两种机制都提供其他一些属性和参数这些属性和参数使开发人员能够控制缓存内容的方式一些VaryByX参数 、缓存内容的位置(Location)和用于设置缓存无效依赖项的功能(SqlDependency) 。
对于传统的输出缓存在您的应用程序中实现该功能时不需要任何其他东西。 OutputCache类型是一个在您的应用程序启动时运行并在遇到页面指令或操作筛选器时开始发挥作用的HttpModule。 收到第一个相关的页面或控制器请求后 ASP.NET将接收生成的内容HTML、
CSS、JavaScript文件等并将各个项目以及过期日期和用于标识相应项目的关键字放入内存缓存中。 过期日期由Duration属性确定关键字则由到页面的路径和必要的VaryBy值的组合确定—例如如果提供了VaryByParam属性则会查询字符串或参数值。 现在请考虑一下以这种方式定义的控制器操作
1. [OutputCache(Duration=20, VaryByParam="vendorState") ]
2. Public ActionResult GetVendorList(string vendorState)
3. {
4. // Action logic here.
5. }
6.
在这种情况下对于vendorState的各个实例例如一个针对德克萨斯州一个针对华盛顿州等等 ASP.NET将在请求该州时分别缓存生成的HTML视图的一个实例。 在这种情况下存储各个实例所使用的关键字将是相关路径和vendorState的组合。
另一方面如果将VaryByParam属性设置为“none” 则ASP.NET将缓存第一次执行GetVendorList的结果并且会向所有后续请求传递相同的缓存版本而不考虑vendorState参数的值是否传入了相应操作。当没有提供VaryByParam值时存储此实例所使用的关键字就是路径。 图1简单描述了此过程。
图1 ASP.NET输出缓存过程
除了用于控制缓存中的项目的生存期的Duration参数以外还有一些VaryBy参数
VaryByParam、 VaryByHeader、 VaryByCustom、 VaryByControl和VaryByContentEncoding用于控制缓存项目的精度可以配置输出缓存以控制缓存内容的位置客户端、服务器或下游代理服务器 。 此外 ASP.NET 2.0引入了一个SqlDependency属性该属性允许开发人员指定页面或控件所依赖的数据库表因此除了在时间上过期以外您的基础源数据的更新也可能会导致缓存项目过期。
尽管.NET Framework 2.0和3.0向默认缓存提供程序中引入了一些增强功能但是提供程序本身仍然没有改变它仍然是一种内存存储并没有用于为您提供您自己的实现的扩展点或方法。 在大多数情况下 内存缓存是一种完全可以接受的方法但有时候 当服务器资源透支且内存不足时它也会削弱站点性能。 此外 当内存确实不足时会导致开发人员无法有效地控制管理缓存资源的方式从而导致默认缓存提供程序机制自动弃用缓存的资源不考虑指定的持续时间 。
ASP.NET中可扩展的输出缓存
.NET Framework 4引入了一种新功能该功能使开发人员能够创建他们自己的输出缓存提供程序并且只需对新的或现有的应用程序及其配置作少量更改便可轻松地将这些提供程序插入其中。 无论他们选择的缓存信息使用的是哪种存储机制例如本地磁盘、相关和非相关数据库、云甚至是分布式缓存引擎如Windows Server AppFabric中所提供的 都可以随意使用这些提供程序。 甚至还可以针对相同应用程序中的不同页面使用多个提供程序。与创建派生自新的System.Web.Caching.OutputCacheProvider抽象类的新类以及覆盖ASP.NET使用缓存项目所需的四种方法一样创建您自己的输出缓存提供程序也很简单。 下面列出了OutputCacheProvider类的框架定义请参见bit. ly/fozTLc以了解更多信息
1. public abstract class OutputCacheProvider : ProviderBase
2. {
3. public abstract object Get(string key) ;
4. public abstract object Add(string key, object entry, DateTime utcExpiry
) ;
5. public abstract void Set(string key, object entry, DateTime utcExpiry) ;
6. public abstract void Remove(string key) ;
7. }
8.
实现了这四种方法后接下来要做的就是向您的web.config添加新提供程序将其指定为默认提供程序并向您的应用程序添加一个OutputCache指令或属性。 我会在介绍如何创建我们自己的输出缓存提供程序使用一个叫做MongoDB的文档数据库时详细说明这些步骤。首先我要介绍一些关于在构建自定义提供程序时使用的工具的背景信息这可能对大家有所帮助。
NoSQL、文档数据库和MongoDB
在过去的数十年中首选应用程序存储机制一直都是关系数据库管理系统(RDBMS) 该系统在表中存储数据和关系。 SQL Server和Oracle都属于RDBMS它们也是目前最受欢迎的商业和开源数据库。
然而并非所有需要存储的问题都适合相同的事务建模。二十世纪九十年代末随着Internet的扩展许多站点都扩大了规模以便托管大量数据很明显关系模型在某些类型的数据密集型应用程序上的性能差强人意。 例如索引大量文档、在高流量站点上向消费者传递网页或向其传递流媒体。
许多公司通过转而使用NoSQL数据库解决了他们不断增加的存储需求 NoSQL数据库是一种不公开SQL接口、 固定架构或预定义关系的轻型数据库。 NoSQL数据库得到了企业例如Google Inc. (BigTable) 、 Amazon.com Inc. (Dynamo)和Facebook 其收件箱搜索的存储量超过了50TB 的广泛使用并且在普及性和使用方面实现了稳定增长。
值得注意的是虽然有些人将NoSQL这个词用作了呼吁停用所有RDBMS的口号但另外一些人则强调同时利用这两类存储的价值。 人们构建NoSQL数据库的目的是解决RDBMS无法解决的一类问题而不是要用它彻底代替这些系统。 有辨别能力的开发人员应该会清楚地了解这两种系统并根据具体情况来利用相应的系统有时甚至会在一个应用程序中混合使用这两种类型的存储。
非常适合使用NoSQL数据库的一种情况就是输出缓存。 NoSQL数据库对于使用瞬态数据或临时数据来说是理想之选而从ASP.NET应用程序缓存的页面当然符合要求。 MongoDB(mongodb.org)是一个常用的NoSQL选项 Shutterfly、 Foursquare、 《纽约时报》等许多公司都在使用这种面向文档的NoSQL数据库。 MongoDB是一个以C++编写的完全开放的源数据库该数据库具有面向几乎所有主要编程语言包括C#的驱动程序。 我们将会把MongoDB用作我们的自定义输出缓存提供程序的存储机制。
使用MongoDB构建自定义的OutputCacheProvider
要开始使用该工具您需要登录mongodb.org下载和安装该工具。mongodb.org/display/DOCS/Quickstart上提供的文档中包含了在Windows、 Mac OS X和Unix上安装MongoDB时所需了解的所有信息。 下载MongoDB并使用外壳程序完成测试后我建议您使用安装目录中的以下命令来将该数据库作为一项服务进行安装请务必以管理员身份运行cmd.exe
C:\Tools\MongoDB\bin>mongod.exe --logpath C:\Tools\MongoDB\Logs--directoryperdb --install
MongoDB将作为一项服务自行安装到您的计算机上并将使用C:\Data\db作为其所有数据库的默认目录。 通过diretoryperdb选项可让MongoDB为您创建的每个数据库创建根目录。运行前一个命令之后键入以下内容以启动服务net start MongoDB
启动并运行MongoDB后您需要在.NET中安装一个驱动程序库以使用MongoDB。 有几个选项可供使用我将使用Sam Corder创建的mongodb-csharp驱动程序
(github.com/samus/mongodb-csharp) 。
我们安装了MongoDB并且在.NET应用程序中有一个可以使用的驱动程序因此现在是时候创建自定义的输出缓存提供程序了。 为了执行这项操作我创建了一个叫做DocumentCache的新类库并添加了两个类 DocumentDatabaseOutputCacheProvider和CacheItem。第一个是我的提供程序它是一个可对抽象OutputCacheProvider进行子类化的公共类。 开始的实现过程如图2所示。
图2起始OutputCacheProvider类
1. public class DocumentDatabaseOutputCacheProvider : OutputCacheProvider
2. {
3. readonly Mongo _mongo;
4. readonly IMongoCollection<CacheItem> _cacheItems;
5.
6. public override object Get(string key)
7. {
8. return null;
9. }
10.
11. public override object Add(string key, object entry, DateTime utcExpiry
)
12. {
13. return null;
14. }
15.
16. public override void Set(string key, object entry, DateTime utcExpiry)
17. {
18. return;
19. }
20.
21. public override void Remove(string key)
22. {
23. return;
24. }
25. }
26.
请注意图2中的第二个私有变量是指CacheItem即我需要在我的项目中创建的另一个类。CacheItem的作用是包含我的输出缓存提供程序与ASP.NET和我的数据库协同工作所需的相关详细信息不过它不是我的提供程序所需的外部对象。 因此我将CacheItem定义为内部类如下所示
1. [Serializable]
2. internal class CacheItem
3. {
4. public string Id { get; set; }
5. public byte[] Item { get; set; }
6. public DateTime Expiration { get; set; }
7. }
8.
Id映射到ASP.NET向我提供的关键字。 您会想到这个关键字是路径和在您的页面指令或操作属性中定义的任何VaryBy条件的组合。 Expiration字段对应于Duration参数而Item属性是要缓存的项目。
我们将通过在DocumentDatabaseOutputCacheProvider类的构造函数中进行一些设置来开始实现我们的提供程序。 由于我们知道ASP.NET在应用程序的整个生存期内仅保留提供程序的一个实例 因此我们可以在构造函数中进行一些设置如下所示
1. readonly Mongo _mongo;
2. readonly IMongoCollection<CacheItem> _cacheItems;
3.
4. public DocumentDatabaseOutputCacheProvider()
5. {
6. _mongo = new Mongo() ;
7. _mongo.Connect() ;
8.
9. var store = _mongo.GetDatabase("OutputCacheDB") ;
10. _cacheItems = store.GetCollection<CacheItem>() ;
11. }
12.
构造函数创建一个Mongo类型的新实例并使用默认位置(localhost)连接到服务器。 之后它向MongoDB请求OutputCacheDB数据库以及一个CacheItem类型的
IMongoCollection。 由于MongoDB是一个架构灵活的数据库因此支持动态创建数据库。第一次调用_mongo.GetDatabase( “OutputCacheDB” )将返回新数据库的一个实例 当第一次插入发生时便会在磁盘上创建该数据库。
现在让我们实现Add方法如图3所示。
图3实现Add方法
1. public override object Add(string key, object entry, DateTime utcExpiry)
2. {
3. key = MD5(key) ;
4. var item = _cacheItems.FindOne(new { _id = key } ) ;
5. if (item != null) {
6. if (item.Expiration.ToUniversalTime() <= DateTime.UtcNow) {
7. _cacheItems.Remove(item) ;
8. } else {
9. return Deserialize(item. Item) ;
10. }
11. }
12.
13. _cacheItems. Insert(new CacheItem
14. {
15. Id = key,
16. Item = Serialize(entry) ,
17. Expiration = utcExpiry
18. } ) ;
19.
20. return entry
21. }
22.
我在各个方法中执行的第一个操作就是对传入的关键字调用MD5方法。 此方法为了保持简洁此处未作详细说明但您可以从在线源代码下载部分中进行查看生成基于ASP.NET向我提供的关键字的、数据库可以识别的MD5哈希。 然后我调用我的
IMongoCollection<CacheItem>类型_cacheItems以便在底层数据库中查询那个关键字。请注意传入FindOne方法中的匿名类型(new {_id = key} ) 。 查询MongoDB主要是通过选择器对象或在文档中指定一个或多个字段以便在数据库中进行匹配的模板文档来完成的。 _id是MongoDB用于存储文档的关键字而按照我所使用的驱动程序的约定该属性会自动映射
到我的CacheItem类的Id属性。 因此 当我保存新的缓存项时正如您在图3中所示的_cacheItems. Insert方法中看到的关键字是使用Id属性进行分配的 MongoDB使用该属性填充记录的内部_id字段。 MongoDB是一种关键字值存储 因此各个CacheItem对象都是使用二进制序列化的JSON进行存储的如下所示
1. { "_id" : ObjectId(Id) , "CacheItem": new CacheItem { Id = key,
Item = entry, Expiration = utcExpiry } }
2.
如果我发现一个CacheItem具有与传入的关键字相同的关键字则我会根据当前的UTC时间来检查该项目是否过期。 如果该项目尚未过期则我会利用私有方法可在在线源代码部分中找到该方法对它进行二进制反序列化并返回现有项目。 此外我向我的存储中插入一个新项目对其进行二进制序列化并返回传入的条目。
向缓存中添加了项目后我便可以添加Get方法该方法将根据关键字查找并返回一个缓存项如果找不到结果则返回nul l 如图4所示。
图4实现Get方法
1. public override object Get(string key)
2. {
3. key = MD5(key) ;
4. var cacheItem = _cacheItems.FindOne(new { _id = key } ) ;
5.
6. if (cacheItem != null) {
7. if (cacheItem.Expiration.ToUniversalTime() <= DateTime.UtcNow) {
8. _cacheItems.Remove(cacheItem) ;
9. } else {
10. return Deserialize(cacheItem. Item) ;
11. }
12. }
13.
14. return null;
15. }
16.
与Add方法一样 Get方法也会检查数据库中存在的项目是否过期如果项目已经过期则会将其删除并返回null。 如果存在的项目尚未过期则会返回该项目。
现在让我们来实现Remove方法该方法可接受一个关键字并从数据库中删除与该关键字相匹配的项目如下所示
1. public override void Remove(string key)
2. {
3. key = MD5(key) ;
4. _cacheItems.Remove(new { _id = key } ) ;
5. }
6.
正如我们的驱动程序为了获得尚不存在的数据库而使用的代码一样如果我们尝试删除数据库中不存在的项目 MongoDB不会报错。 它不会执行任何操作。
根据我们的抽象基类我们仍需要实现最后一个方法—Set方法来获得一个功能性自定义输出缓存提供程序。 我已在图5中使用了该方法。
图5实现Set方法Public Override Void Set(string key, object entry, DateTimeutcExpiry)
1. {
2. key = MD5(key) ;
3. var item = _cacheItems.FindOne(new { _id = key } ) ;
4.
5. if (item != null)
6. {
7. item. Item = Serialize(entry) ;
8. item.Expiration = utcExpiry;
9. _cacheItems.Save(item) ;
10. }
11. else
12. {
13. _cacheItems. Insert(new CacheItem
14. {
15. Id = key,
16. Item = Serialize(entry) ,
17. Expiration = utcExpiry
18. } ) ;
19. }
20. }
21.
总体来看 Add方法似乎与Set方法是相同的但它们的预期实现之间却存在重要区别。 根据关于OutputCacheProvider类的MSDN库文档(bit. ly/fozTLc) 自定义提供程序的Add方法应在缓存中查找与指定关键字匹配的值如果存在这样的值则不对缓存执行任何操作并返回保存的项目。 如果不存在该项目 Add应插入该项目。
另一方面 Set方法应始终将其值保存在缓存中如果不存在该项目则插入该项目如果存在则覆盖该项目。 在关于Add的图3和关于Set的图5中您会注意到这两种方法都是按指定方式执行的。
实现这四种方法后现在我们可以运行我们的提供程序了。
在ASP.NET MVC中使用MongoDB OutputCacheProvider编译完我们的自定义提供程序后我们可以通过几行配置将此提供程序添加到任何ASP.NET应用程序中。 添加对包含此提供程序的程序集的引用后将以下文本添加到您的web.config文件的<system.web>部分
<caching>
<outputCache defaultProvider="DocumentDBCache">
<providers>
<add name="DocumentDBCache"type="DocumentCache.DocumentDatabaseOutputCacheProvider, DocumentCache" />
</providers>
</outputCache>
</caching>
<providers>元素定义您想添加到应用程序中的所有自定义提供程序并为每个提供程序定义名称和类型。 由于您可以在一个应用程序中包含多个自定义提供程序 因此您还希望指定defaultProvider属性如我在之前的代码段中的操作一样。
我的示例应用程序是一个带有CustomersController的简单ASP.NET MVC站点。 该控制器中有一个称为TopCustomers的操作它返回我的业务的顶级客户列表。 此信息经过复杂计算并在我的SQL Server数据库中进行多次数据库查询得出且每小时仅更新一次。 因此它是缓存的理想候选项。 所以我在我的操作中添加了一个OutputCache属性如下所示
1. [OutputCache(Duration = 3600, VaryByParam = "none") ]
2. public ActionResult TopCustomers()
3. {
4. var topCustomers = _repository.GetTopCustomers() ;
5. return View(topCustomers) ;
6. }
7.
现在如果我运行站点并导航到我的TopCustomers页面那么我的自定义提供程序便会执行。首先将调用我的Get方法但是由于此页面还未缓存 因此不会返回任何内容。 接下来控制器操作将执行并返回TopCustomers视图如图6所示。
官方网站:点击访问王小玉网络官网活动方案:买美国云服务器就选MF.0220.CN 实力 强 强 强!!!杭州王小玉网络 旗下 魔方资源池 “我亏本你引流活动 ” mf.0220.CNCPU型号内存硬盘美国CERA机房 E5 2696v2 2核心8G30G总硬盘1个独立IP19.9元/月 续费同价mf.0220.CN 购买湖北100G防御 E5 2690v2 4核心4G...
RAKsmart商家一直以来在独立服务器、站群服务器和G口和10G口大端口流量服务器上下功夫比较大,但是在VPS主机业务上仅仅是顺带,尤其是我们看到大部分主流商家都做云服务器,而RAKsmart商家终于开始做云服务器,这次试探性的新增美国硅谷机房一个方案。月付7.59美元起,支持自定义配置,KVM虚拟化,美国硅谷机房,VPC网络/经典网络,大陆优化/精品网线路,支持Linux或者Windows操作...
IMIDC发布了6.18大促销活动,针对香港、台湾、日本和莫斯科独立服务器提供特别优惠价格最低月付30美元起。IMIDC名为彩虹数据(Rainbow Cloud),是一家香港本土运营商,全线产品自营,自有IP网络资源等,提供的产品包括VPS主机、独立服务器、站群独立服务器等,数据中心区域包括香港、日本、台湾、美国和南非等地机房,CN2网络直连到中国大陆。香港服务器 $39/...