HTTP 缓存相关的响应头信息

响应头信息有:

  1. Cache-Control // 控制缓存
  2. Expire // 控制缓存
  3. ETag // 校验缓存用
  4. Last-Modified // 校验缓存用

Cache-Control

Cache-Control 是用来控制缓存的,比如缓存要不要开启,缓存时长是多少,要不要每次请求缓存前都需要校验等等。ETagLast-Modified 是用于缓存失效后的校验,它们都需要对应的请求头一起使用。

通常 Cache-Control 会搭配 ETagLast-Modified 一起使用用于更好的控制缓存,当然也可以只使用其中一个。

Expires

Expires 头信息返回的值是一个时间日期,在这个日期之后响应被视为已过期。

If there is a Cache-Control header with the max-age or s-maxage directive in the response, the Expires header is ignored.

从这里的定义可以看到它的优先级比 Cache-Control 低,如果响应有 Cache-Control 头信息则 Expires 头信息会被忽略。

这是因为 Cache-ControlETag HTTP 1.1 新增加的响应头,ExpiresLast-Modified HTTP 1.1 之前的响应头。至于新加的响应头的好处其中之一是控制更精确(Last-Modified 只到秒级别)。

ETag

ETag 是响应内容某个指定版本的一个标识符,通常可以用文件的 hash 或者某个固定规则的算法,比如文章最后修改时间的 MD5 等。

它的规范如下,W 开头规范是弱校验用的,这里就不展开了

ETag: W/"<etag_value>"
ETag: "<etag_value>"

ETag 另一个典型的用法是缓存未更改的资源

  1. 客户端再次请求一个指定的资源地址,这个资源缓存已过期
  2. 客户端会通过一个 If-None-Match 请求头发送上次响应的 ETag 的值
  3. 服务端会比较客户端发送的这个 ETag 值 和资源当前版本的 ETag
  4. 如果这两个值相等则会响应一个 304 Not Modified 状态码且没有响应内容
  5. 客户端在接收到 304 响应后就知道上次缓存的内容还是新鲜的,可以继续使用的

Last-Modified

The Last-Modified response HTTP header contains the date and time at which the origin server believes the resource was last modified. It is used as a validator to determine if a resource received or stored is the same. Less accurate than an ETag header, it is a fallback mechanism. Conditional requests containing If-Modified-Since or If-Unmodified-Since headers make use of this field.

ETag and Last-Modified

ETagLast-Modified 是用来校验缓存用的,可以理解成缓存的一个ID,它们都有对应请求头,If-None-MatchIf-Modified-Since,服务端在发出响应之前会根据请求的头信息去匹配计算,比如请求头有 If-None-Match: 'ak1017',服务端在响应之前会计算档次响应的 ETag 和这个值相等就会响应一个 304 没有内容的响应,客户端在接收到 304 之后就会继续使用上次缓存的内容。达到节省流量,提升响应时间。

Last-modified and If-Modified-Since

this is resopone header and request header, they match each other,

ETag and If-None-Match

一个请求命中缓存会响应的状态

  1. 304 no data
  2. 200 disk cache
  3. 200 memory cache

示例一

下面是一个响应头同时包含 Cache-ControlLast-Modified 的例子

// 使用 node + http 实现的部分代码
if (req.url === '/css.css') {
    res.setHeader('Cache-Control', 'max-age=160')
    res.setHeader('Last-Modified', 'Wed, 23 Dec 2020 12:35:35 GMT')
    res.setHeader('Content-Type', 'text/css; charset=UTF-8')

    if (req.headers['if-modified-since']) {
      res.statusCode = 304
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }

下面的访问测试都是基于 chrome 浏览器 87.0.4280.88

No Name Status Type Size
第一次 css.css 200 OK stylesheet 77.6 kB 77.4 kB
第二次 css.css 200 OK stylesheet (memory cache) 77.4 kB
第三次,缓存过期 css.css 304 Not Modified stylesheet 202 B 77.4 kB

示例二

下面是一个响应头同时包含 Cache-ControlETag 的例子

if (req.url === '/etag.css') {
    let etag = 'ak307'

    res.setHeader('Cache-Control', 'max-age=45')
    res.setHeader('Content-Type', 'text/css; charset=UTF-8')
    res.setHeader('ETag', etag)
    if (req.headers['if-none-match'] === etag) {
      res.statusCode = 304
      res.write('here is bad')
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }
No Name Status Type Size
第一次 etag.css 200 OK stylesheet 77.6 kB 77.4 kB
第二次 etag.css 200 OK stylesheet (memory cache) 77.4 kB
第三次,缓存过期 etag.css 304 Not Modified stylesheet 174 B 77.4 kB

这里的再次请求有时会遇到响应 200 但响应内容没有的情况
some | etag.css | 200 OK | stylesheet | 174 B 77.4 kB

设置 Cache-Control 后浏览器再次请求资源时候如果缓存没有过期则就直接使用缓存,在缓存过期后会向服务端发起请求,这时接收到一个 304 客户端继续使用上次的缓存内容并更新了缓存有效期。

示例三

下面是一个响应头只有 ETag 的例子

if (req.url === '/etag2.css') {
    let etag = 'ak308'

    res.setHeader('Content-Type', 'text/css; charset=UTF-8')
    res.setHeader('ETag', etag)
    if (req.headers['if-none-match'] === etag) {
      res.statusCode = 304
      res.write('here is bad')
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }
No Name Status Type Size
第一次 etag2.css 200 OK stylesheet 5.8 kB 5.6kB
第二次 etag2.css 304 Not Modified stylesheet 146 B 77.4 kB
第N次 etag2.css 304 Not Modified stylesheet 146 B 77.4 kB

这里不设置 Cache-Control 浏览器默认也会把资源存入缓存,只是每次请求都会向远端服务器发起校验,然后接收到响应 304 继续使用上次的响应内容。

示例四

下面是一个响应头只有 Last-Modified 的例子

if (req.url === '/style-no-cache.css') {
    const last = 'Wed, 23 Dec 2020 12:35:35 GMT'
    // res.setHeader('Cache-Control', 'max-age=45')
    res.setHeader('Last-Modified', last)
    res.setHeader('Content-Type', 'text/css; charset=UTF-8')
    res.removeHeader('date')

    if (req.headers['if-modified-since'] === last) {
      console.log('last modified enter 304')
      res.statusCode = 304
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }
No Name Status Type Size
第一次 style-no-cache.css 200 OK stylesheet 77.6 kB 77.4 kB
第二次 style-no-cache.css 200 OK stylesheet (memory cache) 77.4 kB
第N次 style-no-cache.css 200 OK stylesheet (memory cache) 77.4 kB

相比较只有 ETag 响应头的,这里只有 Last-Modified 响应头,再次请求都是直接使用缓存的内容,在测试中没发现资源有向远端发起校验,这里也没有设置对应的缓存时长,why?(这里只在设置的 Last-Modified 时间是已过期的情况,如果 Last-Modified 没有过期在过期之前访问还是会响应 304,只是在后续多次访问后会变成 memory cache

示例五

下面是一个响应头包含 ExpiresLast-Modified 的例子

if (req.url === '/style-expires.css') {
    const last = 'Wed, 23 Dec 2020 12:35:35 GMT'
    res.setHeader('Expires', 'Sun, 10 Jan 2020 09:03:00 GMT')
    res.setHeader('Last-Modified', last)
    res.setHeader('Content-Type', 'text/css; charset=UTF-8')

    if (req.headers['if-modified-since'] === last) {
      res.statusCode = 304
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }
No Name Status Type Size
第一次 style-expires.css 200 OK stylesheet 77.6 kB 77.4 kB
第二次 style-expires.css 200 OK stylesheet (memory cache) 77.4 kB
第三次,超过 Expires 时间 style-expires.css 304 Not Modified stylesheet 215 B 77.4 kB

这里可以看到设置 ExpiresCache-Control 有同样的效果

示例六

下面是一个响应头设置 Cache-Control: no-storeETag 的例子

if (req.url === '/style-no-store.css') {
    let etag = 'ak857'

    res.setHeader('Cache-Control', 'no-store')
    res.setHeader('Content-Type', 'text/css; charset=UTF-8')
    res.setHeader('ETag', etag)
    if (req.headers['if-none-match'] === etag) {
      res.statusCode = 304
      res.write('here is bad')
      res.end()
      return
    }

    let css = fs.readFileSync('./css.css')
    res.write(css)
    res.end()
    return
  }
No Name Status Type Size
第一次 style-no-store.css 200 OK stylesheet 77.6 kB 77.4 kB
第N次 style-no-store.css 200 OK stylesheet 77.6 kB 77.4 kB

设置为 no-store 的资源,浏览器端就不会启用缓存,每次都成远端获取最新的内容,且请求头不会追加对应的 If-None-Match

浏览器什么情况下会把请求的资源存入缓存?
只要请求到的响应头开启缓存( Cache-Control: max-age=<number>Expires)或包含 ETag 响应头,或者即使没有设置 Cache-Control 浏览器都会把资源默认存入缓存,除非明确设置 Cache-Control: no-store 不需要浏览器存储缓存。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容