浅析script标签中的defer与async属性


Posted in Javascript onNovember 30, 2016

一、前言

看到的前辈写的代码如下

<script src="#link("xxxx/xx/home/home.js")" type="text/javascript" async defer></script>

竟然同时有asyncdefer属性,心想着肯定是前辈老司机的什么黑科技,两个一块儿肯定会发生什么神奇化学反应,于是赶紧怀着一颗崇敬的心去翻书翻文档,先复习一下各自的定义。

二、调查一番

先看看asyncdefer各自的定义吧,翻开红宝书望远镜,是这么介绍的

2.1 defer

这个属性的用途是表明脚本在执行时不会影响页面的构造。也就是说,脚本会被延迟到整个页面都解析完毕后再运行。因此,在<script>元素中设置defer属性,相当于告诉浏览器立即下载,但延迟执行。

HTML5规范要求脚本按照它们出现的先后顺序执行,因此第一个延迟脚本会先于第二个延迟脚本执行,而这两个脚本会先于DOMContentLoaded事件执行。在现实当中,延迟脚本并不一定会按照顺序执行,也不一定会在DOMContentLoad时间触发前执行,因此最好只包含一个延迟脚本。

2.2 async

这个属性与defer类似,都用于改变处理脚本的行为。同样与defer类似,async只适用于外部脚本文件,并告诉浏览器立即下载文件。但与defer不同的是,标记为async的脚本并不保证按照它们的先后顺序执行。

第二个脚本文件可能会在第一个脚本文件之前执行。因此确保两者之间互不依赖非常重要。指定async属性的目的是不让页面等待两个脚本下载和执行,从而异步加载页面其他内容。

概括来讲,就是这两个属性都会使script标签异步加载,然而执行的时机是不一样的。引用segmentfault上的一个回答中的一张图

浅析script标签中的defer与async属性

蓝色线代表网络读取,红色线代表执行时间,这俩都是针对脚本的;绿色线代表 HTML 解析。

也就是说async是乱序的,而defer是顺序执行,这也就决定了async比较适用于百度分析或者谷歌分析这类不依赖其他脚本的库。从图中可以看到一个普通的<script>标签的加载和解析都是同步的,会阻塞DOM的渲染,这也就是我们经常会把<script>写在<body>底部的原因之一,为了防止加载资源而导致的长时间的白屏,另一个原因是js可能会进行DOM操作,所以要在DOM全部渲染完后再执行。

2.3 really?

然而,这张图(几乎是百度搜到的唯一答案)是不严谨的,这只是规范的情况,大多数浏览器在实现的时候会作出优化。

来看看chrome是怎么做的

《WebKit技术内幕》:

      1、当用户输入网页URL的时候,WebKit调用其资源加载器加载该URL对应的网页。

      2、加载器依赖网络模块建立连接,发送请求并接受答复。

      3、WebKit接收到各种网页或者资源的数据,其中某些资源可能是同步或异步获取的。

      4、网页被交给HTML解释器转变成一系列的词语(Token)。

      5、解释器根据词语构建节点(Node),形成DOM树。

      6、如果节点是JavaScript代码的话,调用JavaScript引擎解释并执行。

      7、JavaScript代码可能会修改DOM树的结构。

      8、如果节点需要依赖其他资源,例如图片、CSS、视频等,调用资源加载器来加载他们,但是他们是异步的,不会阻碍当前DOM树的继续创建;如果是JavaScript资源URL(没有标记异步方式),则需要停止当前DOM树的创建,直到JavaScript的资源加载并被JavaScript引擎执行后才继续DOM树的创建。

所以,通俗来讲,chrome浏览器首先会请求HTML文档,然后对其中的各种资源调用相应的资源加载器进行异步网络请求,同时进行DOM渲染,直到遇到<script>标签的时候,主进程才会停止渲染等待此资源加载完毕然后调用V8引擎对js解析,继而继续进行DOM解析。我的理解如果加了async属性就相当于单独开了一个进程去独立加载和执行,而defer是和将<script>放到<body>底部一样的效果。

三、实验一发

3.1 demo

为了验证上面的结论我们来测试一下

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <title>Document</title>
 <link href="http://libs.baidu.com/bootstrap/3.0.3/css/bootstrap.css" rel="stylesheet">
 <link href="http://cdn.staticfile.org/foundation/6.0.1/css/foundation.css" rel="stylesheet">
 <script src="http://lib.sinaapp.com/js/angular.js/angular-1.2.19/angular.js"></script>
 <script src="http://libs.baidu.com/backbone/0.9.2/backbone.js"></script>
 <script src="http://libs.baidu.com/jquery/2.0.0/jquery.js"></script>
</head>
<body>
 ul>li{这是第$个节点}*1000
</body>
</html>

一个简单的demo,从各个CDN上引用了2个CSS3个JS,在body里面创建了1000个li。通过调整外部引用资源的位置和加入相关的属性利用chrome的Timeline进行验证。

3.2 放置在<head>内

浅析script标签中的defer与async属性

异步加载资源,但会阻塞<body>的渲染会出现白屏,按照顺序立即执行脚本

3.3 放置在<body>底部

浅析script标签中的defer与async属性

异步加载资源,等<body>中的内容渲染完毕后且加载完按顺序执行JS

3.3 放置在<head>头部并使用async

浅析script标签中的defer与async属性

异步加载资源,且加载完JS资源立即执行,并不会按顺序,谁快谁先上

3.4 放置在<head>头部并使用defer

浅析script标签中的defer与async属性

异步加载资源,在DOM渲染后之后再按顺序执行JS

3.5 放置在<head>头部并同时使用async和defer

浅析script标签中的defer与async属性

表现和async一致,开了个脑洞,把这两个属性交换一下位置,看会不会有覆盖效果,结果发现是一致的 = =、

综上,在webkit引擎下,建议的方式仍然是把<script>写在<body>底部,如果需要使用百度谷歌分析或者不蒜子等独立库时可以使用async属性,若你的<script>标签必须写在<head>头部内可以使用defer属性

四、 兼容性

那么,揣摩一下前辈的心理,同时写上的原因是什么呢,兼容性?

上caniuse,async在IE<=9时不支持,其他浏览器OK;defer在IE<=9时支持但会有bug,其他浏览器OK;现象在这个issue里有描述,这也就是“望远镜”里建议只有一个defer的原因。所以两个属性都指定是为了在async不支持的时候启用defer,但defer在某些情况下还是有bug。

The defer attribute may be specified even if the async attribute is specified, to cause legacy Web browsers that only support defer (and not async) to fall back to the defer behavior instead of the synchronous blocking behavior that is the default.

五、结论

其实这么讲来,最稳妥的办法还是把<script>写在<body>底部,没有兼容性问题,没有白屏问题,没有执行顺序问题,高枕无忧,不要搞什么deferasync的花啦~

目前只研究了chrome的webkit的渲染机制,Firefox和IE的有待继续研究,图片和CSS以及其他外部资源的渲染有待研究。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流。

Javascript 相关文章推荐
分享20款好玩的jQuery游戏
Apr 17 Javascript
js中有关IE版本检测
Jan 04 Javascript
javascript延时加载之defer测试
Dec 28 Javascript
实现JavaScript的组成----BOM和DOM详解
May 18 Javascript
JavaScript中的数组遍历forEach()与map()方法以及兼容写法介绍
May 19 Javascript
jquery根据一个值来选中select下的option实例代码
Aug 29 Javascript
AngularJS中的缓存使用
Jan 11 Javascript
vue利用better-scroll实现轮播图与页面滚动详解
Oct 20 Javascript
利用nvm管理多个版本的node.js与npm详解
Nov 02 Javascript
Vue中通过属性绑定为元素绑定style行内样式的实例代码
Apr 30 Javascript
js实现点击选项置顶动画效果
Aug 25 Javascript
全局安装 Vue cli3 和 继续使用 Vue-cli2.x操作
Sep 08 Javascript
JavaScript 字符串常用操作小结(非常实用)
Nov 30 #Javascript
js实现表单提交后不重新刷新当前页面
Nov 30 #Javascript
html判断当前页面是否在iframe中的实例
Nov 30 #Javascript
vue.js实现表格合并示例代码
Nov 30 #Javascript
浅谈jquery的html方法里包含特殊字符的处理
Nov 30 #Javascript
Javascript 闭包详解及实例代码
Nov 30 #Javascript
jQuery特殊符号转义的实现
Nov 30 #Javascript
You might like
PHP中文处理 中文字符串截取(mb_substr)和获取中文字符串字数
2011/11/10 PHP
header中Content-Disposition的作用与使用方法
2012/06/13 PHP
PHP 常用的header头部定义汇总
2015/06/19 PHP
WordPress中调试缩略图的相关PHP函数使用解析
2016/01/07 PHP
ThinkPHP3.2.3实现分页的方法详解
2016/06/03 PHP
利用php获得flv视频长度的实例代码
2017/10/26 PHP
JQuery的ajax获取数据后的处理总结(html,xml,json)
2010/07/14 Javascript
JS 获取浏览器和屏幕宽高等信息代码
2014/03/31 Javascript
jquery合并表格中相同文本的相邻单元格
2015/07/17 Javascript
js 声明数组和向数组中添加对象变量的简单实例
2016/07/28 Javascript
js学习笔记之事件处理模型
2016/10/31 Javascript
微信小程序 实现拖拽事件监听实例详解
2016/11/16 Javascript
详解为Angular.js内置$http服务添加拦截器的方法
2016/12/20 Javascript
ES6 javascript的异步操作实例详解
2017/10/30 Javascript
详解如何使用PM2将Node.js的集群变得更加容易
2017/11/15 Javascript
laypage+SpringMVC实现后端分页
2019/07/27 Javascript
js神秘的电报密码 哈弗曼编码实现
2019/09/10 Javascript
node.js中module模块的功能理解与用法实例分析
2020/02/14 Javascript
javascript设计模式 ? 享元模式原理与用法实例分析
2020/04/15 Javascript
django简单的前后端分离的数据传输实例 axios
2020/05/18 Javascript
JS实现简易贪吃蛇游戏
2020/08/24 Javascript
[41:13]完美世界DOTA2联赛PWL S2 Forest vs Rebirth 第一场 11.20
2020/11/20 DOTA
python使用Queue在多个子进程间交换数据的方法
2015/04/18 Python
Python脚本修改阿里云的访问控制列表的方法
2019/03/08 Python
Python 微信爬虫完整实例【单线程与多线程】
2019/07/06 Python
在Matplotlib图中插入LaTex公式实例
2020/04/17 Python
Python 实现简单的客户端认证
2020/07/29 Python
KEETSA环保床垫:更好的睡眠,更好的生活!
2016/11/24 全球购物
大学生简历中个人的自我评价
2013/10/06 职场文书
《在家里》教后反思
2014/03/01 职场文书
学校出纳员岗位职责
2014/03/18 职场文书
企业道德讲堂实施方案
2014/03/19 职场文书
暑假社会实践心得体会
2014/09/02 职场文书
小学校长个人总结
2015/03/03 职场文书
Python如何快速找到多个字典中的公共键(key)
2022/04/29 Python
Python OpenGL基本配置方式
2022/05/20 Python