Cache-control 我就是我 2021-06-26 16:06 295阅读 0赞 `Cache-Control` 是最重要的规则。这个字段用于指定所有缓存机制在整个`请求/响应`链中必须服从的指令。这些指令指定用于阻止缓存对请求或响应造成不利干扰的行为。这些指令通常覆盖默认缓存算法。缓存指令是单向的,即请求中存在一个指令并不意味着响应中将存在同一个指令。 网页的缓存是由`HTTP`消息头中的“`Cache-control`”来控制的,常见的取值有`private、no-cache、max-age、must-revalidate`等,默认为`private`。其作用根据不同的重新浏览方式分为以下几种情况。 **表 1. 常用 `Cache-control` 值** <table> <thead> <tr> <th align="left">Cache-directive</th> <th align="left">说明</th> </tr> </thead> <tbody> <tr> <td align="left">public</td> <td align="left">所有内容都将被缓存(客户端和代理服务器都可缓存)</td> </tr> <tr> <td align="left">private</td> <td align="left">内容只缓存到私有缓存中(仅客户端可以缓存,代理服务器不可缓存)</td> </tr> <tr> <td align="left">no-cache</td> <td align="left">必须先与服务器确认返回的响应是否被更改,然后才能使用该响应来满足后续对同一个网址的请求。因此,如果存在合适的验证令牌 (ETag),no-cache 会发起往返通信来验证缓存的响应,如果资源未被更改,可以避免下载。</td> </tr> <tr> <td align="left">no-store</td> <td align="left">所有内容都不会被缓存到缓存或 Internet 临时文件中</td> </tr> <tr> <td align="left">must-revalidation/proxy-revalidation</td> <td align="left">如果缓存的内容失效,请求必须发送到服务器/代理以进行重新验证</td> </tr> <tr> <td align="left">max-age=xxx (xxx is numeric)</td> <td align="left">缓存的内容将在 xxx 秒后失效, 这个选项只在HTTP 1.1可用, 并如果和Last-Modified一起使用时, 优先级较高</td> </tr> </tbody> </table> **不同的情形下的浏览器响应** 表 2 表明在不同的情形下,浏览器是将请求重新发送到服务器还是使用缓存的内容。 **表 2. 对 `Cache-control` 值的浏览器响应** <table> <thead> <tr> <th align="left">Cache-directive</th> <th align="left">打开一个新的浏览器窗口</th> <th align="left">在原窗口中单击 Enter 按钮</th> <th align="left">刷新</th> <th align="left">单击 Back 按钮</th> </tr> </thead> <tbody> <tr> <td align="left">public</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">private</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">第一次,浏览器重新发送请求到服务器;此后,浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">no-cache/no-store</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器重新发送请求到服务器</td> </tr> <tr> <td align="left">must-revalidation/proxy-revalidation</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">第一次,浏览器重新发送请求到服务器;此后,浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">max-age=xxx (xxx is numeric)</td> <td align="left">在 xxx 秒后,浏览器重新发送请求到服务器</td> <td align="left">在 xxx 秒后,浏览器重新发送请求到服务器</td> <td align="left">浏览器重新发送请求到服务器</td> <td align="left">在 xxx 秒后,浏览器重新发送请求到服务器</td> </tr> </tbody> </table> `Cache-Control`是关于浏览器缓存的最重要的设置,因为它覆盖其他设置,比如 `Expires` 和 `Last-Modified`。另外,由于浏览器的行为基本相同,这个属性是处理跨浏览器缓存问题的最有效的方法。 **失效** `Expires` 头部字段提供一个日期和时间,响应在该日期和时间后被认为失效。失效的缓存条目通常不会被缓存(无论是代理缓存还是用户代理缓存)返回,除非首先通过原始服务器(或者拥有该实体的最新副本的中介缓存)验证。(注意:`cache-control max-age` 和 `s-maxage` 将覆盖 `Expires` 头部。) `Expires` 字段接收以下格式的值:“`Expires: Sun, 08 Nov 2009 03:37:26 GMT`”。如果查看内容时的日期在给定的日期之前,则认为该内容没有失效并从缓存中提取出来。反之,则认为该内容失效,缓存将采取一些措施。 不同操作下不同浏览器的响应 表 3-6 表明针对不同用户操作的不同浏览器的行为。 **表 3. 当用户打开一个新的浏览器窗口时的失效操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容没有失效</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">内容失效</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 4. 当用户在原始浏览器窗口中单击 `Enter` 按钮时的失效操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容没有失效</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> </tr> <tr> <td align="left">内容失效</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 5. 当用户按 `F5` 键刷新页面时的失效操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容没有失效</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> </tr> <tr> <td align="left">内容失效</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 6. 当用户单击 `Back` 或 `Forward` 按钮时的失效操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容没有失效</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">内容失效</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> 注意:所有浏览器都假定为使用默认设置运行。 **`Last-Modified/E-Tag`** `Last-Modified` 实体头部字段值通常用作一个缓存验证器。简单来说,如果实体值在 `Last-Modified` 值之后没有被更改,则认为该缓存条目有效。`ETag` 响应头部字段值是一个实体标记,它提供一个 “不透明” 的缓存验证器。这可能在以下几种情况下提供更可靠的验证:不方便存储修改日期;HTTP 日期值的 `one-second` 解决方案不够用;或者原始服务器希望避免由于使用修改日期而导致的某些冲突。 不同的浏览器有不同的配置行为。 表 7-10 表明针对不同用户操作的不同浏览器的行为。 **表 7. 当用户打开一个新的浏览器窗口时的 `Last-Modified E-Tag` 操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容自上次访问以来没有被修改</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> </tr> <tr> <td align="left">内容自上次访问以来已经被修改</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 8. 当用户在原始浏览器窗口中单击 `Enter`按钮时的 `Last-Modified E-Tag` 操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容自上次访问以来没有被修改</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> </tr> <tr> <td align="left">内容自上次访问以来已经被修改</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 9. 当用户按 F5 键刷新页面时的 `Last-Modified E-Tag` 操作** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容自上次访问以来没有被修改</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 304</td> </tr> <tr> <td align="left">内容自上次访问以来已经被修改</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> **表 10. 没有缓存设置且用户单击 `Back` 或 `Forward` 按钮** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">内容自上次访问以来没有被修改</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> </tr> <tr> <td align="left">内容自上次访问以来已经被修改</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器呈现来自缓存的页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> 注意:所有浏览器都假定使用默认设置运行。 不进行任何缓存相关设置 如果您不定义任何缓存相关设置,则不同的浏览器有不同的行为。有时,同一个浏览器在相同的情形下每次运行时的行为都是不同的。情况可能很复杂。另外,有些不该缓存的内容如果被缓存,将会导致安全问题。 不同的浏览器有不同的行为。 表 11 展示了不同的浏览器行为。 **表 11. 没有缓存设置且用户打开一个新的浏览器窗口** <table> <thead> <tr> <th align="left"></th> <th align="left">Firefox 3.5</th> <th align="left">IE 8</th> <th align="left">Chrome 3</th> <th align="left">Safari 4</th> </tr> </thead> <tbody> <tr> <td align="left">打开一个新页面</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> <tr> <td align="left">在原始窗口中单击 Enter 按钮</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器呈现来自缓存的页面。</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> <tr> <td align="left">按 F5 键刷新</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> <tr> <td align="left">单击 Back 或 Forward 按钮</td> <td align="left">浏览器呈现来自缓存的页面。</td> <td align="left">浏览器呈现来自缓存的页面。</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> <td align="left">浏览器重新发送请求到服务器。返回代码是 200</td> </tr> </tbody> </table> 注意:所有浏览器都假定使用默认设置运行。 ## 关键结论 ## 最后, 概括下关键的结论: <table> <thead> <tr> <th align="left">操作</th> <th align="left">说明</th> </tr> </thead> <tbody> <tr> <td align="left">打开新窗口</td> <td align="left">如果指定<code>cache-control</code>的值为<code>private、no-cache、must-revalidate</code>,那么打开新窗口访问时都会重新访问服务器。而如果指定了<code>max-age</code>值,那么在此值内的时间里就不会重新访问服务器,例如:<code>Cache-control: max-age=5</code> 表示当访问此网页后的5秒内不会去再次访问服务器.</td> </tr> <tr> <td align="left">在地址栏回车</td> <td align="left">如果值为<code>private</code>或<code>must-revalidate</code>,则只有第一次访问时会访问服务器,以后就不再访问。如果值为<code>no-cache</code>,那么每次都会访问。如果值为<code>max-age</code>,则在过期之前不会重复访问。</td> </tr> <tr> <td align="left">按后退按扭</td> <td align="left">如果值为<code>private、must-revalidate、max-age</code>,则不会重访问,而如果为<code>no-cache</code>,则每次都重复访问.</td> </tr> <tr> <td align="left">按刷新按扭</td> <td align="left">无论为何值,都会重复访问.</td> </tr> </tbody> </table>
相关 Android Okhttp缓存:精细化每一个Request的CacheControl缓存控制策略(二) Android Okhttp缓存:精细化每一个Request的CacheControl缓存控制策略(二) 之前我写的附录文章1,只是简单的使用缺省的方法实现Okh 客官°小女子只卖身不卖艺/ 2024年02月18日 16:01/ 0 赞/ 64 阅读
还没有评论,来说两句吧...