AspNet MVC 缓存

服务端缓存技术

  • 请求域内的缓存

  每个Asp.Net请求都会在Asp.Net框架中创建一个新的System.Web.HttpContext对象(HttpContext对象封装有关个别 HTTP 请求的所有 HTTP 特定的信息)。在HttpContext对象的一个属性就是HttpContext.Items,这个属性也会存在于请求的整个周期,以HttpContext.Items为媒介可以解除各个组件之间的耦合。Items是一个IDictionary的类型可以直接通过HttpContext.Item["key"]=value的形式获取。Items可以用户获取Asp.Net请求过程中IHttpModule和IHttpHandler之间组织和共享数据的键值对。

  • 用户域内的缓存

  用户域的缓存 也就是平常使用的HttpContext.Session,仅仅是当前用户的缓存,不能实现跨用户共享数据。Aspnet的会话状态可以再多个请求之间共享数据,记得这个只能是当前用户可以得到共享数据。会话状态空闲一段时间后会过期,当然可以通过Web.config的<system.web>节点下修改<seesionState timeout="30" />进行设置,当然还有其他参数可以进行设置。

  • 应用程序域内的缓存

  平常使用的HttpContext.Application,为当前 HTTP 请求获取 HttpApplicationState 对象。那么就是HttpApplicationState管理这应用程序级别的数据。微软官方的解释:“启用 ASP.NET 应用程序中多个会话和请求之间的全局信息共享。”,一样也是一个键值对类型。存储在HttpApplicationState的数据生命空间域托管网站的IIS工作进程一样长,因为IIS不是Asp.NET管理着工作线程的生命周期。HttpApplicationState也是不是可靠地数据存储方式。

  • Asp.Net缓存

  平常比较经常用的HttpContext.Cache进行访问,与Application一样也是应用程序域的数据。Cache存储的数据不好限制在单个请求或者用户会话范围内。HttpContext.Cache与Application还是有区别的,Application可以扩工作进程的访问。HttpContext.Cache以HttpRuntime.Cache的实现。我们知道HttpContext是通过封装个别的Http请求信息所得到的只能应用于Web应用上,而HttpRuntime.Cache是获取当前应用程序的Cache对象,可以非web应用。 使用Cache对象的时候可以设置过期时间(有两种方式其一设置间隔时间,用于指定缓存数据在最后一次访问之后多长过期,其二指定固定时间设置失效。对于Cache添加数据的时候还有一个缓存依赖数据,我们可以通过指定依赖项,来决定其生命周期缓存依赖策略

依赖类型   定义
Aggregate(聚合) 提供多个依赖(通过System.Web.Caching.AggregateCache)。只有删除所有依赖项才能删除该缓存记录
Custom(自定义)    缓存数据依赖一个继承自System.Web.Caching.Cache的自定义类,例如,可以创建一个web服务缓存依赖,当查询某个值时,就删除缓存数据
File(文件)   缓存数据依赖某个外部的文件,删除文件就会删除该数据
Key(键)  缓存数据依赖应用程序缓存里另外一个缓存数据(通过缓存的key关联),删除依赖的缓存数据就会删除该数据。
SQL 缓存数据依赖MicrosoftSQLServer某个表的数据变化,当表更新时,缓存数据就会删除。

 

  • 输出缓存

  平常用于输出缓存主要是OutputCache属性,当使用OutputCache属性缓存页面,用户访问到页面时,首先会查看服务器的是否有该页面的缓存,若有则直接获取显示出来,而不会执行数据操作生成页面。我们可以对OutputCache属性进行参数的设置比如Duration可以设置缓存的时间等。具体可以查看微软的Msdn。具体的使用如下所示:

[OutputCache(Duration=60,VaryByParam="none"]
public ActionResult Contact()
{
     ViewBag.Message=DateTime.Now.ToString();
     return View()      
}

 这里列出OutputCacheAttribute定义的属性

参数  描述
CacheProfile 使用的输出缓存策略的名字
Duration 缓存内容的生命周期
Enabled 是否启用缓存
Location 缓存地址默认设置为Any,这以为这内容可以缓存到三个地方Web 服务器、任何代理服务器和用户浏览器。
NoStore   是否启用Http Cache-Control
SqlDependency 缓存依赖的数据库和表名称
VaryByContentEncoding 都好分割的字符编码列表,用来区分输入缓存
VaryByCustom 自动以字符串用来区分输出缓存
VaryByHeader 都好分割的HTTP消息头,区分缓存
VaryByParam 通过分割的Post表单或者查询字符串来区分缓存,比如查看某个id下的页面的页面,可以设置该属性为"id"
  • 甜圈圈缓存(donut caching)

甜圈圈缓存是指只是缓存页面的外围内容,允许部分内容动态变化;与之相反的是甜圈圈洞的缓存(donut hole caching)(也就是缓存内部部分内容,而 允许外围内容的更新)。可以通过调用HttpResponse中的一个WriteSubstitution()方法实现甜圈圈缓存。具体的使用如下所示:

public delegate string CacheCallback(HttpContextBase context);

public static object Substitution(this HtmlHelper html, CacheCallback ccb)
{
      html.ViewContext.HttpContext.Response.WriteSubstitution( 
             c=>HttpUtility.HtmlEncode(
                    ccb(new HttpContextWrapper(c)      
             ));      
}

//具体调用
@Html.Substitution(context=>context.User.Identity.Name)

 当然也可以引入MvcDonutCachingNuGet包提供几种高级的缓存扩展方法,也提供了操作方法使用DonutOutputCacheAttribute标记属性。

  •  甜圈圈洞缓存

缓存只是页面中的一小部分,针对某些类别信息是不会变化可以进行缓存。具体的使用如下:

[ChildActionOnly]
[OutputCache(Duration=60)]
public ActionResult CategoriesAction()
{
    ViewBag.Categories=Model.GetCategories();
    return View();
}

//对应的页面CategoriesAction
@foreach(var item in ViewBag.Categories as IEnumerable<Category>)
{
      @Html.ActionLink(item.Name,"category","categories",new{categoryId=category.Id})
}

//其他主页面调用,调用第一次的调用结果缓存起来,下次渲染的时候直接从缓存中获得列表
<body>
@Html.Action("CategoriesAction");
</body>

客户端缓存技术

  •  浏览器缓存

浏览器在本地的资源通过三个机制控制:新的(freshness )验证(validation)和失效的(invalidation)这三者属于Http的一部分,并且定在Http消息头里。

新的允许应答消息直接使用,而不需要经过服务器检查,额可以同被服务器和客户端控制。例如Expires过期的消息头设置了资源过期的时间,而Cache-Control:maxage则停工的是应答消息需要经过多久才刷新。

验证:用来检查陈旧的缓存数据是否依赖有效。如果缓存的应答数据包含Last-Modified消息头,则缓存就可以使用If-Modified-Since消息头来检查内容是否更新过。也可以通过ETag进行更强的控制。

失效:通常是指另一个缓存请求的错误请求

  • AppCache缓存和本地缓存(Html5新特性)【暂不介绍】

 参考:Asp.Net Mvc 4 WEeb 编程

 

posted @ 2014-04-22 22:01  卤鸽  阅读(1117)  评论(3编辑  收藏  举报