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

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

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


当前回答

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

对于。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、代理和其他网络设备也缓存资源,它们经常使用内部规则而不查看实际资源。你对此无能为力。好消息是,它们通常缓存更短的时间框架,比如几秒或几分钟。

其他回答

对于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());
    }

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

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

你只需要:

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

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

[OutputCache(Duration=0)]
public class MyController

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

ASP。NET MVC 5解决方案:

在中央位置缓存防止代码:App_Start/FilterConfig.cs的RegisterGlobalFilters方法:

    public class FilterConfig
    {
        public static void RegisterGlobalFilters(GlobalFilterCollection filters)
        {
            // ...
            filters.Add(new OutputCacheAttribute
            {
                NoStore = true,
                Duration = 0,
                VaryByParam = "*",
                Location = System.Web.UI.OutputCacheLocation.None
            });
        }
    }

一旦你有了它,我的理解是,你可以通过在控制器或视图级别应用不同的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 = "*")]

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

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

如微软文档所述:

响应缓存在ASP。NET核心-没有存储和位置。没有一个