http缓存

缓存相关header

  • Expires

响应头,代表资源的过期时间

  • Cache-Control

请求/响应头,缓存控制字段,精确控制缓存策略

  • If-modified-Since

请求头,资源最近修改时间,由浏览器告诉服务器

  • Last-Modified

响应头,资源最近修改时间,由服务器告诉浏览器

  • Etag

响应头,资源标识,由服务器告诉浏览器

  • If-None-Match

请求头,缓存资源标识,由浏览器告诉服务器

配对使用的字段

  • If-modified-Since和Last-Modified
  • Etag和If-None-Match

从实际场景出发,一点点完善缓存机制,来理解各个字段的意义

做一些约定,方便以后比较。

  • a.js 大小为10KB
  • 请求头约定为1KB
  • 响应头约定为1KB

原始模型

  • 浏览器请求静态资源a.js(请求头1KB)
  • 服务器读取磁盘文件,返回给浏览器。(文件+响应头 11KB)
  • 浏览器再次请求,服务器又重新读取文件,返回给浏览器

消耗的流量与访问次数正相关

浏览增加缓存机制

  • 浏览器第一次请求a.js,缓存a.js到本地磁盘 (1+10+1= 12KB)
  • 浏览器再次请求a.js,直接走浏览器缓存,不再向服务器发起请求 (0KB)

流量与访问次数无关,只有第一次请求会消耗浏览
缺点:服务器a.js更新后,浏览器也拿不到最新的资源

服务器和浏览器约定资源过期时间 Expires

  • 浏览器第一次请求静态资源a.js(1KB)
  • 服务器把文件和文件缓存过期时间发给浏览器
  • 浏览器收到文件,记住了过期时间
  • 在过期时间之前,浏览器再次请求a.js,会直接使用上一次缓存的文件
  • 在过期时间之后,再次请求,会去请求服务器,服务器重新读取文件返回给浏览器,同时告知浏览器新的过期时间。

缺点:缓存过期之后,服务器不管a.js有无变化,都会再次读取a.js返回给浏览器

服务器告诉浏览器资源上次修改时间

  • 浏览器访问a.js
  • 服务器返回a.js时,告诉浏览器a.js的上次修改时间Last-Modified及缓存过期时间Expires
  • 当过期时间之后请求a.js,会带上 If-Modified-Since(等于上一次请求的Last-Modified)
  • 服务器比较请求头里的If-Modified-Since和文件的上次修改时间
    • 如果一致,则告知浏览器可以继续使用本地缓存(304)
    • 如果不一致,则读取文件返回给浏览器,并带上新的 Last-Modified及缓存过期时间Expires

缺点:Expires 过期控制不稳定;Last-Modified 过期时间只能精确到秒(一秒内文件变化频繁,获取不到最新的文件;可能存在文件未变化,但修改时间更新了的情况)。

增加相对时间的控制Cache-Contorl

为了兼容已有缓存方案,同时加入新的缓存方案。除了告诉浏览器Expires,还告诉一个相对时间 Cache-Control:max-age=10秒,含义是10秒内使用缓存到浏览器的a.js。
浏览器先检查Cache-Control,如果有,则以Cache-Control为准,忽略Expires。如果没有Cache-Control则以Expires为准。

Cache-Control的常用设置值:

  • no-cache:不使用本地缓存,使用协商缓存
  • no-store:直接禁止浏览器缓存数据,每次用户请求该资源,都会向服务器发送请求,服务器都重新下载资源
  • public:可以被所有用户缓存,包括终端用户和CDN等中间代理服务器
  • private:只能被终端用户的浏览器缓存,不允许中继缓存服务器对其缓存

增加文件内容对比,ETag 和 If-None-Match

a.js 内容变了,Etag 才变。内容不变,Etag 不变,可以理解为 Etag 是文件内容的唯一 ID。

每次浏览器请求服务器的时候,都带上If-None-Match字段,该字段的值就是上次请求 a.js 时,服务器返回给浏览器的 Etag。

  • 浏览器请求a.js.
  • 服务器返回a.js,同时告诉浏览器过期绝对时间Expires和相对时间Cache-Control:max-age=10),文件上次修改时间Last-Modified以及文件Etag。
  • 10秒内,浏览器再次请求,直接用本地缓存。
  • 10秒后,浏览器再次请求,带上上次修改时间 If-Modified-Since和If-None-Match。
  • 服务器收到请求,发现有If-Modified-Since和If-None-Match。存在If-None-Match,则忽略If-Modified-Since。如果If-None-Match的值跟文件的Etag值相同,则告诉浏览器继续使用本地缓存,否则返回新的文件及其他字段信息。

还存在的问题

无论是Expires还是Cache-Control,在缓存过期之前,浏览器都不会发请求询问服务器,如果在这期间,服务器里的文件发生了变化,浏览器是感知不到的。

可以想象一下我们使用a.js的场景,一般都是输入网址,访问一个html文件,html文件里引入js、css、图片等资源。
所以我们可以设置让html文件不缓存,每次都拿到最新的html。然后如果js文件内容有更新时,更新引用js的地方,可以通过版本号来区分,也可以通过hash值区分。使用webpack打包的话,借助插件可以很方便的处理。

<script src="http://test.com/a.js?version=0.0.1"></script>
<script src="http://test.com/a.【hash值】.js"></script>

强缓存和协商缓存
强缓存是利用Expires和Cache-Control两个字段来控制的。
协商缓存是由服务器来确定缓存资源是否可用,主要通过Etag和If-None-Match、Last-Modified和If-Modified-Since这两组字段来控制。

参考:
面试精选之http缓存
HTTP强缓存和协商缓存
前端缓存最佳实践

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

推荐阅读更多精彩内容

  • 前端面试常问第二大问题是http缓存相关内容。说真的,http缓存相关的细节比较多,并且 http 常用协议版本有...
    wdapp阅读 54评论 0 0
  • 缓存相关 header Expires 响应头,代表该资源的过期时间。 Cache-Control 请求/响应头,...
    mills_han阅读 435评论 0 0
  • 无缓存,原始模型 浏览器请求静态资源 a.js。(请求头:1KB) 服务器读取磁盘文件 a.js,返给浏览器。(1...
    明明你也一样阅读 392评论 0 0
  • 最近在看面试题的时候总会看到有一些关于Http缓存的题目,但是总是一知半解,不甚理解;尤其是Http头信息中有一大...
    竿牍阅读 830评论 0 0
  • 接着上篇《<计算机与网络篇 > web缓存机制》,其细分出来四个缓存机制,而前端工程师能干预的也只有浏览器缓存这一...
    Max_Law阅读 435评论 0 1