引言
在angular-start项目中启用了模块热替换(HMR - Hot Module Replacement)@H_502_4@功能,关于如何在
angular-cli@H_502_4@启用
HRM@H_502_4@,请查看HRM配置
那HMR@H_502_4@是个什么东西呢?
HMR@H_502_4@是
webpack@H_502_4@提供的一个功能,
angular-cli@H_502_4@使用了它,它会在应用程序运行过程中替换、添加或删除模块,而无需重新加载整个页面。主要是通过以下几种方式,来显著加快开发速度:
这一切是如何运行的?
我们先看看具体的效果:
1、启动angular-start@H_502_4@项目,在控制台你可以看到
HRM@H_502_4@已经启用的消息:
2、然后通过浏览器控制台可以看到,第一次加载请求了所有的资源:
3、此时,修改一处代码保存,浏览器并未刷新就自动显示修改之后的效果,再看浏览器控制台只请求了新修改的js:
下面让我们从一些不同的角度观察,以了解HMR@H_502_4@的工作原理……
在应用程序中
通过以下步骤,可以做到在应用程序中置换(swap in and out@H_502_4@)模块:
- 应用程序代码要求 HMR runtime 检查更新
- HMR runtime(异步)下载更新,然后通知应用程序代码
- 应用程序代码要求 HMR runtime 应用更新
- HMR runtime(异步)应用更新
在编译器中
除了普通资源,编译器(compiler@H_502_4@)需要发出
update@H_502_4@,以允许更新之前的版本到新的版本。
update@H_502_4@由两部分组成:
manifest@H_502_4@包括新的编译
hash@H_502_4@和所有的待更新
chunk@H_502_4@目录。每个更新
chunk@H_502_4@都含有对应于此
chunk@H_502_4@的全部更新模块(或一个
flag@H_502_4@用于表明此模块要被移除)的代码。
编译器确保模块ID@H_502_4@和
chunk ID@H_502_4@在这些构建之间保持一致。通常将这些
ID@H_502_4@存储在内存中(例如,使用
webpack-dev-server@H_502_4@时),但是也可能将它们存储在一个
JSON@H_502_4@文件中。
在模块中
HMR@H_502_4@是可选功能,只会影响包含
HMR@H_502_4@代码的模块。举个例子,通过
style-loader@H_502_4@为
style@H_502_4@样式追加补丁。为了运行追加补丁,
style-loader@H_502_4@实现了
HMR@H_502_4@接口;当它通过
HMR@H_502_4@接收到更新,它会使用新的样式替换旧的样式。
类似的,当在一个模块中实现了HMR@H_502_4@接口,你可以描述出当模块被更新后发生了什么。然而在多数情况下,不需要强制在每个模块中写入
HMR@H_502_4@代码。如果一个模块没
HMR@H_502_4@处理函数,更新就会冒泡。这意味着一个简单的处理函数能够对整个模块树
(complete module tree)@H_502_4@进行更新。如果在这个模块树中,一个单独的模块被更新,那么整组依赖模块都会被重新加载。
有关 module.hot 接口的详细信息,请查看HMR API页面。
在HMR Runtime中
对于模块系统的runtime@H_502_4@,附加的代码被发送到
parents@H_502_4@和
children@H_502_4@跟踪模块。在管理方面,
runtime@H_502_4@支持两个方法
check@H_502_4@和
apply@H_502_4@。
check@H_502_4@发送
HTTP@H_502_4@请求来更新
manifest@H_502_4@。如果请求失败,说明没有可用更新。如果请求成功,待更新
chunk@H_502_4@会和当前加载过的
chunk@H_502_4@进行比较。对每个加载过的
chunk@H_502_4@,会下载相对应的待更新
chunk@H_502_4@。当所有待更新
chunk@H_502_4@完成下载,就会准备切换到
ready@H_502_4@状态。
apply@H_502_4@方法将所有被更新模块标记为无效。对于每个无效模块,都需要在模块中有一个更新处理函数,或者在它的父级模块们中有更新处理函数。否则,无效标记冒泡,并也使父级无效。每个冒泡继续直到到达应用程序入口起点,或者到达带有更新处理函数的模块(以最先到达为准)。如果它从入口起点开始冒泡,则此过程失败。
之后,所有无效模块都被(通过dispose@H_502_4@处理函数)处理和解除加载。然后更新当前
hash@H_502_4@,并且调用所有
accept@H_502_4@处理函数。
runtime@H_502_4@切换回闲置状态,一切照常继续。