浏览器加载、渲染和解析过程黑箱简析


Posted in Javascript onNovember 29, 2012

用 Fiddler 监控,在 IE6 下,资源下载顺序为:
浏览器加载、渲染和解析过程黑箱简析

很明显,下载顺序从上到下,文档流中先出现的资源先下载。在 IE8, Safari, Chrome 等浏览器下也类似。

Firefox 对下载顺序做了优化
浏览器加载、渲染和解析过程黑箱简析
Firefox 会将 js, css 提前下载,而将图片等资源延迟到后面下载。

对于渲染,利用 Fiddler 将网速调慢,可以看到 css 下载后会马上渲染到页面,渲染和下载同步进行。js 的解析和运行,也类似。

对于 js 运行,以及页面加载相关事件的触发,特别做了测试。在 Firefox 下,打开测试页面:

[22:13:32.947] HTML Start[22:13:32.947] normal inline script run time[22:13:34.904] normal external script run time[22:13:35.775] [body] normal external script run time[22:13:35.789] [body end] normal external script run time[22:13:35.789] HTML End[22:13:35.791] deferred inline script run time[22:13:35.791] deferred external script run time[22:13:35.793] DOMContentLoaded[22:13:38.144] images[0] onload[22:13:38.328] images[1] onload[22:13:39.105] images[2] onload[22:13:39.105] images[3] onload[22:13:39.106] window.onload

很明显,JS 的运行严格按照文档流中的顺序进行。其中 deferred 的脚本会在最后运行(注:Firefox 3.5 开始支持 defer,而且支持得很完美)。

再来看下 IE8,结果如下:

[22:33:56.806] HTML Start[22:33:56.826] normal inline script run time[22:33:57.786] normal external script run time[22:33:57.812] deferred inline script run time[22:33:57.816] document.readyState = interactive[22:33:57.934] [body] normal external script run time[22:33:58.310] [body end] normal external script run time[22:33:58.310] HTML End[22:33:58.346] deferred external script run time[22:33:58.346] images[0].readyState = loading[22:33:58.346] images[0].readyState = complete[22:33:58.346] images[0] onload[22:33:58.361] doScroll[22:33:58.451] images[1].readyState = loading[22:33:58.479] images[1].readyState = complete[22:33:58.479] images[1] onload[22:33:58.794] images[2].readyState = loading[22:33:58.854] images[2].readyState = complete[22:33:58.854] images[2] onload[22:33:58.876] images[3].readyState = loading[22:33:58.876] images[3].readyState = complete[22:33:58.876] images[3] onload[22:33:58.887] document.readyState = complete[22:33:58.888] window.onload

可以看出,IE8 下,defer 只对 external 脚本有效,对 inline 脚本无效。另,与 DOMContentLoaded 最接近的是 doScroll. 这是 doScroll 被广泛用来模拟 DOMContentLoaded 的原因。小心:仅仅是模拟,细节上并不等价。

还可以得到一个有点意外的结果:放在 body 结束前的脚本,执行时,依旧最好放在 domready 事件中。无论在 Firefox 还是 IE 下,解析到 HTML End 时,并不代表 DOM 可以安全操作,特别是页面比较复杂时。

从上面数据中,也可以看出 YSlow 性能优化法则里,建议将样式置顶和脚本置底的根据。

有兴趣的可以进一步测试动态添加样式和脚本的情形,会稍有不同,但没有特别 surprise.

最后总结下

页面资源的下载顺序是从上到下的,文档流中先出现的资源先下载(注:存在并发,具体请参考 UA Profiler)。当某一样式下载完成时,会立刻渲染到页面(体现了层叠样式表中层叠在渲染时的含义)。当某一脚本下载完成时,也会立刻解析和运行。脚本的运行严格按照文档流中的顺序进行,deferred 的脚本会在正常脚本运行之后运行(Firefox 和 IE 下)。

特别需要留意:脚本运行时,会暂停该脚本之下所有资源的下载(因为脚本可能改变文档流,甚至跳转页面,浏览器的暂停策略是合理的)。要小心内联脚本,经常会阻塞后续下载。

好了,废话不多说。以上结果,建议各位亲自测试,反复测试,疯狂测试,一直到眼花缭乱稀里糊涂恍然大悟继续糊涂为止……

Javascript 相关文章推荐
Jquery Autocomplete 结合asp.net使用要点
Oct 29 Javascript
DOM_window对象属性之--clipboardData对象操作代码
Feb 03 Javascript
jQuery中的$.ajax()方法应用
May 06 Javascript
jQuery延迟加载图片插件Lazy Load使用指南
Mar 25 Javascript
javascript中 try catch用法
Aug 16 Javascript
jQuery插件Validate实现自定义校验结果样式
Jan 18 Javascript
RequireJs的使用详解
Feb 19 Javascript
JS组件系列之JS组件封装过程详解
Apr 28 Javascript
在vue-cli脚手架中配置一个vue-router前端路由
Jul 03 Javascript
JavaScript实现微信红包算法及问题解决方法
Apr 26 Javascript
详解JS浏览器事件循环机制
Mar 27 Javascript
小程序中使用css var变量(使js可以动态设置css样式属性)
Mar 31 Javascript
javascript控制swfObject应用介绍
Nov 29 #Javascript
javascript 保存文件到本地实现方法
Nov 29 #Javascript
jquery连缀语法如何实现
Nov 29 #Javascript
javascript 使td内容不换行不撑开
Nov 29 #Javascript
json原理分析及实例介绍
Nov 29 #Javascript
javascript全局变量封装模块实现代码
Nov 28 #Javascript
Javascript Request获取请求参数如何实现
Nov 28 #Javascript
You might like
什么是短波收听SWL
2021/03/01 无线电
让PHP以ROOT权限执行系统命令的方法
2011/02/10 PHP
PHP判断是手机端还是PC端 PHP判断是否是微信浏览器
2017/03/15 PHP
PHP排序算法之快速排序(Quick Sort)及其优化算法详解
2018/04/21 PHP
PHP+MySQL实现输入页码跳转到指定页面功能示例
2018/06/01 PHP
laravel中的fillable和guarded属性详解
2019/10/23 PHP
网页设计常用的一些技巧
2006/12/22 Javascript
利用javascript移动div层-javascript 拖动层
2009/03/22 Javascript
js类型检查实现代码
2010/10/29 Javascript
基于jquery的图片的切换(以数字的形式)
2011/02/14 Javascript
js获取select默认选中的Option并不是当前选中值
2014/05/07 Javascript
jQuery不兼容input的change事件问题解决过程
2014/12/05 Javascript
JS实现仿google、百度搜索框输入信息智能提示的实现方法
2015/04/20 Javascript
基于socket.io+express实现多房间聊天
2016/03/17 Javascript
Bootstrap导航条学习使用(一)
2017/02/08 Javascript
详解vue-cli中的ESlint配置文件eslintrc.js
2017/09/25 Javascript
修改Nodejs内置的npm默认配置路径方法
2018/05/13 NodeJs
在 Angular6 中使用 HTTP 请求服务端数据的步骤详解
2018/08/06 Javascript
angular5 子组件监听父组件传入值的变化方法
2018/09/30 Javascript
IE9 elementUI文件上传的问题解决
2018/10/17 Javascript
JavaScript 复制对象与Object.assign方法无法实现深复制
2018/11/02 Javascript
vue动态绑定class选中当前列表变色的方法示例
2018/12/19 Javascript
在vue中使用vuex,修改state的值示例
2019/11/08 Javascript
[01:09:19]DOTA2-DPC中国联赛 正赛 VG vs Aster BO3 第二场 2月28日
2021/03/11 DOTA
浅谈Python2.6和Python3.0中八进制数字表示的区别
2017/04/28 Python
python线程池(threadpool)模块使用笔记详解
2017/11/17 Python
33个Python爬虫项目实战(推荐)
2019/07/08 Python
python利用opencv保存、播放视频
2020/11/02 Python
css3实现书本翻页效果的示例代码
2021/03/08 HTML / CSS
建筑工程技术应届生求职信
2013/11/17 职场文书
酒店总经理欢迎词
2014/01/08 职场文书
物业保安员岗位职责制度
2014/01/30 职场文书
领导班子党的群众路线对照检查材料
2014/09/25 职场文书
国博复兴之路观后感
2015/06/02 职场文书
Linux安装Nginx步骤详解
2021/03/31 Servers
mysql拆分字符串作为查询条件的示例代码
2022/07/07 MySQL