前端缓存最佳实践

点击上方“前端开发博客”,选择“设为星标”

回复“2”加入前端群

作者:黑金团队

https://juejin.cn/post/6844903737538920462

前言

缓存,这是一个老生常谈的话题,也常被作为前端面试的一个知识点。本文,重点在与探讨在实际项目中,如何进行缓存的设置,并给出一个较为合理的方案。

强缓存和协商缓存


在介绍缓存的时候,我们习惯将缓存分为强缓存和协商缓存两种。两者的主要区别是使用本地缓存的时候,是否需要向服务器验证本地缓存是否依旧有效。顾名思义,协商缓存,就是需要和服务器进行协商,最终确定是否使用本地缓存。

两种缓存方案的问题点


强缓存

我们知道,强缓存主要是通过http请求头中的Cache-Control和Expire两个字段控制。Expire是HTTP1.0标准下的字段,在这里我们可以忽略。我们重点来讨论的Cache-Control这个字段。一般,我们会设置Cache-Control的值为“public, max-age=xxx”,表示在xxx秒内再次访问该资源,均使用本地的缓存,不再向服务器发起请求。显而易见,如果在xxx秒内,服务器上面的资源更新了,客户端在没有强制刷新的情况下,看到的内容还是旧的。如果说你不着急,可以接受这样的,那是不是完美?然而,很多时候不是你想的那么简单的,如果发布新版本的时候,后台接口也同步更新了,那就gg了。有缓存的用户还在使用旧接口,而那个接口已经被后台干掉了。怎么办?

协商缓存

协商缓存最大的问题就是每次都要向服务器验证一下缓存的有效性,似乎看起来很省事,不管那么多,你都要问一下我是否有效。但是,对于一个有追求的码农,这是不能接受的。每次都去请求服务器,那要缓存还有什么意义。

最佳实践


缓存的意义就在于减少请求,更多地使用本地的资源,给用户更好的体验的同时,也减轻服务器压力。所以,最佳实践,就应该是尽可能命中强缓存,同时,能在更新版本的时候让客户端的缓存失效。在更新版本之后,如何让用户第一时间使用最新的资源文件呢?机智的前端们想出了一个方法,在更新版本的时候,顺便把静态资源的路径改了,这样,就相当于第一次访问这些资源,就不会存在缓存的问题了。

伟大的webpack可以让我们在打包的时候,在文件的命名上带上hash值。

entry:{

main: path.join(__dirname,‘./main.js’),

vendor: [‘react’, ‘antd’]

},

output:{

path:path.join(__dirname,‘./dist’),

publicPath: ‘/dist/’,

filname: ‘bundle.[chunkhash].js’

}

复制代码

综上所述,我们可以得出一个较为合理的缓存方案:

  • HTML:使用协商缓存。

  • CSS&JS&图片:使用强缓存,文件命名带上hash值。

哈希也有讲究


webpack给我们提供了三种哈希值计算方式,分别是hash、chunkhash和contenthash。那么这三者有什么区别呢?

  • hash:跟整个项目的构建相关,构建生成的文件hash值都是一样的,只要项目里有文件更改,整个项目构建的hash值都会更改。

  • chunkhash:根据不同的入口文件(Entry)进行依赖文件解析、构建对应的chunk,生成对应的hash值。

  • contenthash:由文件内容产生的hash值,内容不同产生的contenthash值也不一样。

显然,我们是不会使用第一种的。改了一个文件,打包之后,其他文件的hash都变了,缓存自然都失效了。这不是我们想要的。那chunkhash和contenthash的主要应用场景是什么呢?在实际在项目中,我们一般会把项目中的css都抽离出对应的css文件来加以引用。如果我们使用chunkhash,当我们改了css代码之后,会发现css文件hash值改变的同时,js文件的hash值也会改变。这时候,contenthash就派上用场了。

ETag计算


Nginx

Nginx官方默认的ETag计算方式是为"文件最后修改时间16进制-文件长度16进制"。例:ETag:“59e72c84-2404”

Express

Express框架使用了serve-static中间件来配置缓存方案,其中,使用了一个叫etag的npm包来实现etag计算。从其源码可以看出,有两种计算方式:

  • 方式一:使用文件大小和修改时间

function stattag (stat) {

var mtime = stat.mtime.getTime().toString(16)

var size = stat.size.toString(16)

return ‘"’ + size + ‘-’ + mtime + ‘"’

}

复制代码

  • 方式二:使用文件内容的hash值和内容长度

function entitytag (entity) {

if (entity.length === 0) {

// fast-path empty

return ‘“0-2jmj7l5rSw0yVb/vlWAYkK/YBwk”’

}

// compute hash of entity

var hash = crypto

.createHash(‘sha1’)

.update(entity, ‘utf8’)

.digest(‘base64’)

.substring(0, 27)

// compute length of entity

var len = typeof entity === ‘string’

? Buffer.byteLength(entity, ‘utf8’)

: entity.length

return ‘"’ + len.toString(16) + ‘-’ + hash + ‘"’

}

复制代码

ETag与Last-Modified谁优先


协商缓存,有ETag和Last-Modified两个字段。那当这两个字段同时存在的时候,会优先以哪个为准呢?在Express中,使用了fresh这个包来判断是否是最新的资源。主要源码如下:

function fresh (reqHeaders, resHeaders) {

// fields

var modifiedSince = reqHeaders[‘if-modified-since’]

var noneMatch = reqHeaders[‘if-none-match’]

// unconditional request

if (!modifiedSince && !noneMatch) {

return false

}

// Always return stale when Cache-Control: no-cache

// to support end-to-end reload requests

// https://tools.ietf.org/html/rfc2616#p-14.9.4

var cacheControl = reqHeaders[‘cache-control’]

if (cacheControl && CACHE_CONTROL_NO_CACHE_REGEXP.test(cacheControl)) {

return false

}

// if-none-match

if (noneMatch && noneMatch !== ‘*’) {

var etag = resHeaders[‘etag’]

if (!etag) {

return false

}

var etagStale = true

var matches = parseTokenList(noneMatch)

for (var i = 0; i < matches.length; i++) {

var match = matches[i]

if (match === etag || match === ‘W/’ + etag || ‘W/’ + match === etag) {

etagStale = false

break

}

}

if (etagStale) {

return false

}

}

// if-modified-since
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(资料价值较高,非无偿)

更多面试题

**《350页前端校招面试题精编解析大全》**内容大纲主要包括 HTML,CSS,前端基础,前端核心,前端进阶,移动端开发,计算机基础,算法与数据结构,项目,职业发展等等

资料获取方式:点击蓝色传送门即可获取

部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**

如果你觉得这些内容对你有帮助,可以扫码获取!!(资料价值较高,非无偿)

更多面试题

**《350页前端校招面试题精编解析大全》**内容大纲主要包括 HTML,CSS,前端基础,前端核心,前端进阶,移动端开发,计算机基础,算法与数据结构,项目,职业发展等等

资料获取方式:点击蓝色传送门即可获取

[外链图片转存中…(img-5p1RNKA7-1711685218542)]

  • 16
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值