5

在这里对“过期”标题感到非常困惑!有时它会按预期工作 - 有时不会。

我正在使用以下代码来设置过期标头。请注意,这是在 MVC 自定义属性中使用 ASP.NET 完成的——这在这里并不真正相关——但解释了'filterContext'来自哪里。

HttpCachePolicyBase cache = filterContext.HttpContext.Response.Cache;
TimeSpan cacheDuration = TimeSpan.FromSeconds(Duration);

// my own custom header so we know what time it was
filterContext.HttpContext.Response.AddHeader("CurrentTime", DateTime.Now.ToString());

cache.SetCacheability(HttpCacheability.Public);
cache.SetExpires(DateTime.Now.Add(cacheDuration));
cache.SetMaxAge(cacheDuration);
cache.AppendCacheExtension("must-revalidate, proxy-revalidate");

这有时会给我这样的标题:

Cache-Control: public, must-revalidate, proxy-revalidate, max-age=413
Date: Wed, 18 Feb 2009 05:24:19 GMT
Expires: Wed, 18 Feb 2009 05:21:12 GMT
CurrentTime: 2/17/2009 9:21:12 PM

有时像这样:

Cache-Control: public, must-revalidate, proxy-revalidate, max-age=600
Date: Wed, 18 Feb 2009 05:27:55 GMT
Expires: Wed, 18 Feb 2009 05:27:55 GMT
CurrentTime: 2/17/2009 9:27:55 PM

我正在通过 Fiddler 运行所有内容,并查看何时重新请求内容以及何时来自浏览器缓存。

现在奇怪的是在 IE 中缓存总是按预期工作。我的 ASP.NET MVC 操作方法的链接出现在 Fiddler 中,然后当我再次单击同一链接时,它来自缓存。

但是在 Chrome 中,它有时会,有时不会来自缓存!通过来自缓存,我的意思是没有额外的 HTTP 请求。

例如这样的链接:

 http://ipv4.fiddler:62669/gallery/mainimage/2

将来自 IE 中的缓存,但在 chrome 中返回 200。然后有时在 Chrome 中它确实来自缓存。我已经尝试清空浏览器缓存并再次尝试 - 每次都得到相同的结果。

Chrome 是否正在尝试做一些“聪明”的事情并且只是惨遭失败 - 还是我需要一个额外的标题?

我想知道的是,这是否与我的Expires标题日期实际上永远不会在未来这一事实有关。如果我查看谷歌托管的 jQuery 文件的标题,我会看到标题如下(2010 年到期 - 未来一年)。

Cache-Control: public, max-age=31536000
Date: Wed, 18 Feb 2009 05:44:53 GMT
Expires: Thu, 18 Feb 2010 05:44:53 GMT

Expires 实际上不应该在未来吗?

根据HTTP 规范

如果响应包含 Expires 标头和 max-age 指令,则 max-age 指令将覆盖 Expires 标头,即使 Expires 标头更具限制性。对于给定的响应,此规则允许源服务器向 HTTP/1.1(或更高版本)缓存提供比 HTTP/1.0 缓存更长的到期时间。如果某些 HTTP/1.0 缓存不正确地计算年龄或过期时间,这可能很有用,可能是由于时钟不同步。

因此,即使“过期”与当前时间相同,Chrome 似乎也应该尊重 max-age 指令,但它似乎并没有这样做。

4

2 回答 2

3

我在 ASP.NET MVC 源代码中发现了以下内容:

 public virtual void RenderView(ViewContext viewContext) {
        // TODO: Remove this hack. Without it, the browser appears to always load cached output
        viewContext.HttpContext.Response.Cache.SetExpires(DateTime.Now);
        ViewUserControlContainerPage containerPage = new ViewUserControlContainerPage(this);
        // Tracing requires Page IDs to be unique.
        ID = Guid.NewGuid().ToString();

        RenderViewAndRestoreContentType(containerPage, viewContext);
    }

所以这解释了为什么我的Expires标题总是是当前时间。但是我真的不认为这是让 Chrome 绊倒的原因,因为我创建了如下最简单的页面,Chrome 仍然很高兴回到服务器并给了我 200

public partial class test : System.Web.UI.Page
{
    protected void Page_Load(object sender, EventArgs e)
    {
        TimeSpan cacheDuration = TimeSpan.FromSeconds(33);
       var cache = Response.Cache;

        cache.SetCacheability(HttpCacheability.Public);
        cache.SetExpires(DateTime.Now.Add(cacheDuration));
        cache.SetMaxAge(cacheDuration);
        cache.AppendCacheExtension("must-revalidate, proxy-revalidate");
    }
}
于 2009-02-18T06:32:15.150 回答
1

我几乎得出的结论是,Chrome 在缓存方面做了一些非常糟糕的事情。

我将它简化到尽可能低的级别——从 Google 的服务器获取 jQuery。

我输入:

http://ajax.googleapis.com/ajax/libs/jquery/1.3.1/jquery.min.js 

到 Chrome,Fiddler 提出了以下200请求:

Requests started at:    22:58:00:7756
Responses completed at: 22:58:03:5020
Total Sequence time:    00:00:02.7263880
DNS Lookup time:    531ms
TCP/IP Connect time:    63ms

RESPONSE CODES
--------------
HTTP/200:   1

标题如下(注意过期是今天之后的 1 年):

Cache-Control: public, max-age=31536000
Date: Wed, 18 Feb 2009 06:58:01 GMT
Expires: Thu, 18 Feb 2010 06:58:01 GMT
Vary: Accept-Encoding

然后我等了几秒钟,然后在同一个选项卡中按回车键。Fiddler 提出了另一个 **200* 请求:

Requests started at:    22:58:09:2516
Responses completed at: 22:58:12:3999
Total Sequence time:    00:00:03.1482360

RESPONSE CODES
--------------
HTTP/200:   1

标题是:

Cache-Control: public, max-age=31536000
Date: Wed, 18 Feb 2009 06:58:09 GMT
Expires: Thu, 18 Feb 2010 06:58:09 GMT
Vary: Accept-Encoding

显然这不是我所期待的。

- 两个请求的接受编码相同。

的 - 第三个请求给了我一个 304

这是在我从未做过任何开发的新安装的 Chrome 上 - 而且我只是第一次安装了 Fiddler。

我等不及有人向我解释这一点。现在我放弃了——我认为我的缓存和过期代码很好。此外,ASP.NET MVC 似乎强制将 Expires 设置为当前时间。这显然不是我的谷歌示例中的一个因素。

我认为 Chrome 太聪明了,这一定是一个错误——我使用的是 1.0.154.48 版本。

于 2009-02-18T07:14:46.343 回答