Webpack实现持久缓存

作者: dev双皮奶 | 来源:发表于2018-01-20 13:30 被阅读32次

持久缓存

使用webpack构建工程的时候,我们往常会把功能不同的代码打包到不同的包里(如lib,vendor,业务代码)。 而持久缓存的目的就是每一次更新线上代码的时候,尽可能使内容未做更改的模块的名字和之前保持一致。

使用webpack实现持久缓存主要需要解决:

1. webpack runtime代码分离

2. 稳定moduleID

3. 稳定chunkID

基本配置

webpack有提供2种hash命名的方式

1.    [hash]: 整次build生成一个唯一的hash值,赋给所有生成的文件。

2.    [chunkhash]: 每个文件会根据其内容生成不同的hash值.

显然,为了将文件名和内容相关联,应该使用chunkhash。

webpack配置 业务代码 build结果

这样配置虽然可以实现持久缓存,但是把所有的代码都打到了一起(vendor, 业务代码等)。

Webpack-Runtime

使用CommonsChunkplugin将vendor单独打包:

vendor打包配置 build结果

现在如果我对业务代码进行更改:

更新后的业务代码

按道理vendor已经单独打包,改变业务代码并不应该改变vendor的hash值,然而

新生成的vendor名变了

原因是CommonsChunkplugin把vendor单独打出来的时候,还会将webpack自己生成的一部分runtime代码一起打进vendor. 如下图,runtime里牵扯到chunkid等容易频繁变更的元素,所以当业务代码发生变化的时候,runtime代码也会变。

这个问题也容易解决,再写一层CommonsChunkplugin把runtime代码单独打出来(CommonsChunkplugin会把runtime的代码打到配置指定的最后一个chunk里):

单独打包runtime代码 vendor,业务代码,runtime代码都单独打包

现在再更改业务代码,业务代码和manifest的内容会变,vendor文件的内容不会受影响。

ModuleID

然而还没完,当我们在业务代码里增加一个entry,vendor的hash值又发生了变化。

增加一个entry 增加entry后build结果

造成这个问题的原因是当我们加入一个新entry的时候,会在业务代码里新增一个module,而webpack默认会依次用整数给这些module命名。比如说当只有一个entry的时候,业务代码里定义了module: 0,1,2,3。vendor里定义了module 4,5。增加一个entry后:业务代码里会定义: 0,1,2,3,4。 vendor里定义module 5,6.

一个entry时的vendor代码头部 二个entry时的vendor代码头部

可见,要生成稳定的chunkhash值,首先必须解决moduleID的问题。

NamedModulesPlugin & HashedModuleIdsPlugin

NamedModulesPlugin:使用文件的相对路径作为moduleID

相对路径代替整数

不过也带来2个问题: 

1.用相对路径代替数字,文件变大了

2.相对路径暴露了

HashedModuleIdsPlugin :主要就是为了解决以上2个问题,它对相对路径进行一个md5的摘要,不仅避免文件过大,也隐藏了路径。

ChunkID

有时候一些模块可能在页面初始化的时候并用不到,可能会在之后的过程中(比如用户点击事件)才会用到,这一类模块可以通过动态import()来引入。

动态引入foo 动态引入后的build结果

可以看到,动态引入的代码会被单独打包到chunks/里的文件里,而且vendor的值再次发生了改变。造成这个的原因是和moduleID类似,webpack默认使用整数作为chunkID,并且异步加载的chunk会先被赋值。 也就是说在没有动态引入之前,vendor的chunkID是0,动态引入之后,vendor文件的chunkID变为了1,所以造成了内容变化。

NamedChunksPlugin:这个插件会用chunk的字符串name代替整数作为chunkID。不过要注意的是,动态生成的chunk并没有名字,所以需要手动给取个名字。

配置

相关文章

  • Webpack实现持久缓存

    持久缓存 使用webpack构建工程的时候,我们往常会把功能不同的代码打包到不同的包里(如lib,vendor,业...

  • webpack 持久化缓存

    前言 什么是持久化缓存,为什么做持久化缓存? webpack 如何做持久化缓存? 持久化缓存 首先我们需要去解释一...

  • webpack5新特性

    1 持久化缓存 缓存在webpack5中默认开启,缓存默认是在内存里,但可以对cache进行设置cache: { ...

  • 使用Webpack设计一个所有项目适用的分包配置

    本文是对《设计一个无懈可击的浏览器缓存》文章的延申,其中应该有以两个系列的文章: Webpack生成能够持久缓存的...

  • Varnish 接口缓存

    varnish基础概念 可以基于内存缓存,也可以在磁盘上缓存,但是就算存放在磁盘上,也不能实现持久缓存 只要进成本...

  • Java面试——Redis和缓存

    参考资料: Redis的存储类型及底层实现 ?? Redis持久化数据和缓存怎么做扩容? 如果Redis被当做缓存...

  • webpack持久化实践

    webpack持久化实践

  • Redis入门(特点 应用 数据结构 函数)

    Redis特点(C 实现的) 内存存储,查询速度快,经常用作缓存服务器 支持数据持久化,重启会加载持久化的数据,支...

  • 07-Redis实现缓存

    新建一个工程编写整合 Redis 实现缓存案例,工程如图: 可以看到添加了三个依赖组件,包含持久层的数据库,缓存的...

  • hibernate3

    1持久类三种状态 oid Hibernate缓存 一级缓存session 持久态自动更新数据库

网友评论

    本文标题:Webpack实现持久缓存

    本文链接:https://www.haomeiwen.com/subject/bpazoxtx.html