大白话讲webpack热更新

webpack热更新,也称热替换,英文缩写HMR(Hot Module Replacement)。ok,介绍完定义,那么HMR的作用是什么呢?答案是让web项目可以在开发环境下,每次更改文件后无需重刷浏览器,即可看见更改的效果。嗯,了解完这些后,让我来抛出以下问题:

  • 怎么打造一个高效的开发环境?
  • 我在开发环境中咋没见过webpack构建的资源呢?
  • 浏览器怎么知道要更新哪些内容?
  • 为什么有时候更改代码是热更新、有时候浏览器刷新呢?

相信实际开发中,我们很少去想过以上的问题,“管他呢能用就行”“项目进度赶得飞起”是很多人的常态,就算想去了解也被动不动整篇都讲源码的文章吓跑。本篇文章希望通过大白话,让读者能够了解热更新是什么,帮开发做了什么。当然了,大白话讲的意思就是不要出现整段整段的代码,尽量争取简单易懂。

先讲下webpack与开发环境

实际开发中我们都会用现成的脚手架去创建项目,用成熟脚手架创建的项目自然也就带上了一系列开发环境配置,比如热更新。那如果我们不用脚手架呢?从零开始会是什么样,看以下几个阶段:

  1. 手工劳作阶段:代码更新 >> 手动webpack构建 >> 生成打包资源 >> 手动刷新浏览器看效果。这一阶段全靠手动操作。
  2. watch阶段,使用webpack --watch模式启动项目:代码更新 >> 自动webpack构建 >> 生成打包资源 >> 手动刷新浏览器。这一阶段每次更新代码后,不用手动构建一次了,webpack自动构建代码,提效了一部分。
  3. devServer阶段:在第2阶段的基础上,增加了一个静态资源服务器
  4. 热更新阶段:代码更新 >> 触发webpack自动构建 >> 浏览器自动替换变更模块

通过这几个阶段演进,可以看到开发的效率越来越高,从这里我们看到了一个高效的开发环境,但中间依然有几个疑点,比如阶段3之前,我们可以看到构建后有资源文件的产出(默认输出到根目录下的dist目录),但热更新阶段却并没有看到每次构建产生的包,那构建资源到哪去了呢?其次,浏览器怎么知道要更新哪一块内容呢?在回答这些问题之前,让我们先看看一些名词定义。

先了解一些名词

  • 关于webpack-dev-server(以下称WDS):一个基于node.js和webpack的简易服务器,限开发时使用。WDS包含三个核心功能,分别是提供一个简易的web服务器webpack-dev-middleware以及webpack-dev-server/client

  • 关于webpack-dev-middleware:一个中间件,用于与webpack的compiler对象绑定,在dev-server启动时会调用,限开发使用。有以下三个特性:

    • 通过watch mode,监听资源变更,然后自动打包。
    • 打包后的文件存入内存中,而不是磁盘。
    • 支持HMR
  • 关于webpack-dev-server/client:与服务端进行一个websocket连接,并对服务端的消息作出响应及消息传递。

  • 关于HMR.runtimeHotModuleReplacementPlugin插件提供,用于在浏览器端请求更新后的资源以及进行热替换操作。

  • 关于webpack/hot/dev-server:根据webpack-dev-server/client传递的消息,决定是进行live reload浏览器刷新还是热更新

热更新流程

相信通过对一些名词的解释,有的人大体已经猜到热更新的流程,接下来大体走一遍步骤。

  1. 在webpack watch模式下更新代码,webpack会自动进行构建。顺带提一下,以webpack-dev-server运行默认watch:true
  2. WDS与webpack之间进行交互,实际上是由WDS的中间件webpack-dev-middleware与webpack进行接口交互,webpack-dev-middleware对webpack暴露的api进行监控并告诉webpack将资源打包到内存中,保存为一个Javascript对象。
  3. 如果在webpack中配置了devServer.watchContentBase: true,WDS会监听配置中的静态文件的变化,并进行live reload浏览器刷新,注意是直接刷新。静态文件什么意思呢?就比如.html文件等entry入口找不到的文件。
  4. 通过WDS里的sockjs在浏览器端和服务器之间建立websocket长连接。这一步的目的是将webpack编译打包的各个阶段状态告知浏览器,让浏览器端可以根据这些消息进行不同的操作,主要传递的还是新模块的hash值。这里涉及浏览器端中webpack-dev-server/client,问题来了,这东西什么时候在浏览器端代码中的呢?先存个疑。
  5. 这一步是判断步骤,webpack/hot/dev-server会根据上一步中webpack-dev-server/client传递的状态及dev-server的配置,决定是刷新浏览器还是进行热更新,如果是热更新则进入下一步,如果是刷新浏览器则流程到此结束。
  6. HMR.runtime登场,它接收到webpack-dev-server/client传递的新模块hash值,然后通过JsonpMainTemplate.runtime向server端发送Ajax请求,请求返回一个json,包含了所有要更新的模块hash值,然后该模块再通过jsonp请求,获取到最新的模块代码。
  7. 更新模块,HMR会对新旧模块进行对比,决定是否更新模块,决定更新后,检查模块之间的依赖关系,更新模块的同时更新模块间的依赖引用。
  8. 如果上一步HMR更新失败,则回退到live reload的操作。

补充

以下内容是对本篇文章上述部分的补充

问题:webpack/Node.js如何监听文件变化?

查一下Node.js的文档,找到两个监听文件变化的API:fs.watch | fs.watchFile
在webpack通过配置watch可以开启监听文件变化,原理如下:
在webpack中,监听一个文件发生变化的原理是定时获取该文件的最后编辑时间,如果发现该时间与上一次保存的时间不一致则认为该文件发生了变化。其中,我们可以在配置项watchOptions.poll中设置轮询的间隔时间。
在发现一个文件发生变动时,并不会马上通知监听者,而是先缓存起来,收集一段时间后,再通知。这段时间的配置可以在watchOptions.aggregateTimeout配置,默认值300ms

问题:HMR.runtime及webpack-dev-server/client何时注入到浏览器代码(即bundle)中?

HMR.runtime:是由HotModuleReplacementPlugin插入,现在默认在webpack/lib/HotModuleReplacement.runtime
webpack-dev-server/client:WDS修改了webpack配置中的entry属性,在里面添加了webpack-dev-server/client的代码,这样在最后生成的bundle.js文件中就有了这一部分的代码了。

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