defer和async是script标签的两个属性,用于在不阻塞页面文档解析的前提下,控制脚本的下载和执行。
我们先来了解页面文档的渲染机制
- 浏览器通过HTTP协议请求服务器,获取HMTL文档并开始从上到下解析,构建DOM。
- 在构建DOM过程中,如果遇到外联的样式声明和脚本声明,则暂停文档解析,创建新的网络连接,并开始下载样式文件和脚本文件。
- 样式文件下载完成后,构建CSSDOM;脚本文件下载完成后,解释并执行,然后继续解析文档构建DOM 。
- 完成文档解析后,将DOM和CSSDOM进行关联和映射,最后将视图渲染到浏览器窗口。
在这个过程中,JS脚本文件的下载和执行是与文档解析同步进行的,如果JS出现堵塞,将会影响页面渲染,出现白屏、FOUC等现象,影响用户体验。因此我们需要使用defer和async来控制脚本异步加载。
defer和async的作用如下:
- defer:用于开启新的线程下载脚本文件,并使脚本在文档解析完成后执行。
- async:用于异步下载脚本文件,下载完毕立即解释执行代码。
下图可以更清楚地阐述defer和async的执行以及和DOMContentLoaded、load事件的关系:
绿色线是页面解析
蓝色线是JS下载
红色线是JS执行
我们可以明显看到defer和async的脚本下载都是异步进行的,而两者区别是defer要在页面解析完成后执行脚本,async是下载完脚本立刻执行脚本,同时async会影响脚本的解析。
下面是defer和async基本语法:
<script async src="script.js"></script>
<script defer src="script.js"></script>
关于defer我们需要注意下面几点:
- defer只适用于外联脚本,如果script标签没有指定src属性,只是内联脚本,不要使用defer;
- 如果有多个声明了defer的脚本,则会按顺序下载和执行 ;
- defer脚本会在DOMContentLoaded和load事件之前执行。
关于async,也需要注意以下几点:
- 只适用于外联脚本,这一点和defer一致;
- 如果有多个声明了async的脚本,其下载和执行也是异步的,不能确保彼此的先后顺序;
- async会在load事件之前执行,但并不能确保与DOMContentLoaded的执行先后顺序 。
网友评论