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

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

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


当前回答

MVC中的输出缓存

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

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

其他回答

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

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

如微软文档所述:

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

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

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 = "*")]

为了确保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)来刷新它。