防止在ASP中缓存。NET MVC中使用属性的特定操作

我有一个ASP。NET MVC 3应用程序。这个应用程序通过jQuery请求记录。jQuery回调控制器动作,返回JSON格式的结果。我还不能证明这一点,但我担心我的数据可能会被缓存。

我只希望缓存应用于特定的操作,而不是所有的操作。

是否有一个属性可以放在操作上以确保数据不会被缓存?如果不是,我如何确保浏览器每次都获得一组新的记录,而不是缓存的记录集?

196264 次浏览

为了确保JQuery不会缓存结果,在你的ajax方法中,放入以下内容:

$.ajax({
cache: false
//rest of your ajax setup
});

或者为了防止在MVC中缓存,我们创建了自己的属性,你也可以这样做。下面是我们的代码:

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]
public sealed class NoCacheAttribute : ActionFilterAttribute
{
public override void OnResultExecuting(ResultExecutingContext filterContext)
{
filterContext.HttpContext.Response.Cache.SetExpires(DateTime.UtcNow.AddDays(-1));
filterContext.HttpContext.Response.Cache.SetValidUntilExpires(false);
filterContext.HttpContext.Response.Cache.SetRevalidation(HttpCacheRevalidation.AllCaches);
filterContext.HttpContext.Response.Cache.SetCacheability(HttpCacheability.NoCache);
filterContext.HttpContext.Response.Cache.SetNoStore();


base.OnResultExecuting(filterContext);
}
}

然后用[NoCache]装饰你的控制器。或者,你可以把属性放在你继承控制器的基类的类上(如果你有一个),就像我们这里:

[NoCache]
public class ControllerBase : Controller, IControllerBase

如果你需要一些动作是不可缓存的,你也可以用这个属性来装饰它们,而不是装饰整个控制器。

如果你的类或动作在浏览器中呈现时没有NoCache,而你想检查它是否工作,请记住,在编译更改后,你需要在浏览器中执行“硬刷新”(Ctrl+F5)。在你这样做之前,你的浏览器会保留旧的缓存版本,不会用“正常刷新”(F5)来刷新它。

您可以使用内置的缓存属性来防止缓存。

对于.net Framework: [OutputCache(NoStore = true, Duration = 0)]

对于。net Core: [ResponseCache(NoStore = true, Duration = 0)]

请注意,不可能强制浏览器禁用缓存。你能做的最好的事情就是提供大多数浏览器都认可的建议,通常以标题或元标签的形式。这个装饰器属性将禁用服务器缓存,并添加这个头:Cache-Control: public, no-store, max-age=0。它不添加元标记。如果需要,可以在视图中手动添加。

此外,JQuery和其他客户端框架将试图欺骗浏览器不使用资源的缓存版本,通过添加一些东西到url,如时间戳或GUID。这可以有效地使浏览器再次请求资源,但并不能真正阻止缓存。

最后一点。您应该知道,资源也可以缓存在服务器和客户机之间。ISP、代理和其他网络设备也缓存资源,它们经常使用内部规则而不查看实际资源。你对此无能为力。好消息是,它们通常缓存更短的时间框架,比如几秒或几分钟。

在控制器动作中,将以下行附加到标题

    public ActionResult Create(string PositionID)
{
Response.AppendHeader("Cache-Control", "no-cache, no-store, must-revalidate"); // HTTP 1.1.
Response.AppendHeader("Pragma", "no-cache"); // HTTP 1.0.
Response.AppendHeader("Expires", "0"); // Proxies.

你只需要:

[OutputCache(Duration=0)]
public JsonResult MyAction(

或者,如果你想为整个控制器禁用它:

[OutputCache(Duration=0)]
public class MyController

尽管在这里的评论中有争论,但这足以禁用浏览器缓存——这会导致ASP。Net发出响应头,告诉浏览器文档立即过期:

OutputCache Duration=0 Response Headers: max-age=0, s-maxage=0

下面是mattytommo提出的NoCache属性,通过使用Chris Moschini的答案来简化:

[AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]
public sealed class NoCacheAttribute : OutputCacheAttribute
{
public NoCacheAttribute()
{
this.Duration = 0;
}
}

MVC中的输出缓存

[OutputCache(NoStore = true, Duration = 0, Location="None", VaryByParam = "*")]


OR
[OutputCache(NoStore = true, Duration = 0, VaryByParam = "None")]


对于MVC6 (DNX),没有System.Web.OutputCacheAttribute

注意:设置NoStore Duration参数时不考虑。可以为第一次注册设置初始持续时间,并使用自定义属性覆盖该持续时间。

但是我们有Microsoft.AspNet.Mvc.Filters.ResponseCacheFilter

 public void ConfigureServices(IServiceCollection services)
...
services.AddMvc(config=>
{
config.Filters.Add(
new ResponseCacheFilter(
new CacheProfile() {
NoStore=true
}));
}
...
)

可以使用自定义属性覆盖初始过滤器

    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method)]
public sealed class NoCacheAttribute : ActionFilterAttribute
{
public override void OnResultExecuting(ResultExecutingContext filterContext)
{
var filter=filterContext.Filters.Where(t => t.GetType() == typeof(ResponseCacheFilter)).FirstOrDefault();
if (filter != null)
{
ResponseCacheFilter f = (ResponseCacheFilter)filter;
f.NoStore = true;
//f.Duration = 0;
}


base.OnResultExecuting(filterContext);
}
}

这是一个用例

    [NoCache]
[HttpGet]
public JsonResult Get()
{
return Json(new DateTime());
}

为了防止浏览器缓存(包括Internet Explorer 11), Asp。Net MVC核心的正确属性值是:

[ResponseCache(Location = ResponseCacheLocation.None, NoStore = true)]

如微软文档所述:

ASP. js中的响应缓存。NET核心-没有存储和位置。没有一个< / >

ASP。NET MVC 5解决方案:

  1. 在中心位置缓存防止代码:App_Start/FilterConfig.csRegisterGlobalFilters方法:
    public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
// ...
filters.Add(new OutputCacheAttribute
{
NoStore = true,
Duration = 0,
VaryByParam = "*",
Location = System.Web.UI.OutputCacheLocation.None
});
}
}
  1. 一旦你有了它,我的理解是,你可以通过在ControllerView级应用不同的OutputCache指令来覆盖全局过滤器。对于常规控制器来说
[OutputCache(NoStore = true, Duration = 0, Location=System.Web.UI.ResponseCacheLocation.None, VaryByParam = "*")]

或者如果它是ApiController,它就是

[System.Web.Mvc.OutputCache(NoStore = true, Duration = 0, Location = System.Web.UI.OutputCacheLocation.None, VaryByParam = "*")]