`
JavaSam
  • 浏览: 955172 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

javascript模块化加载又称非阻塞加载或异步加载

 
阅读更多

关键词:异步加载(async loading),延迟加载(lazy loading),延迟执行(lazy execution),async 属性, defer 属性

一、同步加载与异步加载的形式

1. 同步加载

我们平时最常使用的就是这种同步加载形式:

<script src="http://yourdomain.com/script.js"></script> 

同步模式,又称阻塞模式,会阻止浏览器的后续处理,停止了后续的解析,因此停止了后续的文件加载(如图像)、渲染、代码执行。

 js 之所以要同步执行,是因为 js 中可能有输出 document 内容、修改dom、重定向等行为,所以默认同步执行才是安全的。

以前的一般建议是把<script>放在页面末尾</body>之前,这样尽可能减少这种阻塞行为,而先让页面展示出来。

简单说:加载的网络 timeline 是瀑布模型,而异步加载的 timeline 是并发模型。

2. 常见异步加载(Script DOM Element

(function() {
     var s = document.createElement('script');
     s.type = 'text/javascript';
     s.async = true;
     s.src = 'http://yourdomain.com/script.js';
     var x = document.getElementsByTagName('script')[0];
     x.parentNode.insertBefore(s, x);
 })();


异步加载又叫非阻塞,浏览器在下载执行 js 同时,还会继续进行后续页面的处理。

这种方法是在页面中<script>标签内,用 js 创建一个 script 元素并插入到 document 中。这样就做到了非阻塞的下载 js 代码。

async属性是HTML5中新增的异步支持,见后文解释,加上好(不加也不影响)。

此方法被称为 Script DOM Element 法,不要求 js 同源。

js代码包裹在匿名函数中并立即执行的方式是为了保护变量名泄露到外部可见,这是很常见的方式,尤其是在 js 库中被普遍使用。

例如 Google Analytics  Google+ Badge 都使用了这种异步加载代码:

(function() {
     var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true;
     ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js';
     var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s);
 })();
 

(function()

    {var po = document.createElement("script");
    po.type = "text/javascript"; po.async = true;po.src = "https://apis.google.com/js/plusone.js";
    var s = document.getElementsByTagName("script")[0];
    s.parentNode.insertBefore(po, s);
 })();
 

但是,这种加载方式在加载执行完之前会阻止 onload 事件的触发,而现在很多页面的代码都在 onload 时还要执行额外的渲染工作等,所以还是会阻塞部分页面的初始化处理。

3. onload 时的异步加载

(function() {
     function async_load(){
         var s = document.createElement('script');
         s.type = 'text/javascript';
         s.async = true;
         s.src = 'http://yourdomain.com/script.js';
         var x = document.getElementsByTagName('script')[0];
         x.parentNode.insertBefore(s, x);
     }
     if (window.attachEvent)
         window.attachEvent('onload', async_load);
     else
         window.addEventListener('load', async_load, false);
 })();
 

这和前面的方式差不多,但关键是它不是立即开始异步加载 js ,而是在 onload 时才开始异步加载。这样就解决了阻塞 onload 事件触发的问题。

补充:DOMContentLoaded  OnLoad 事件

DOMContentLoaded : 页面(document)已经解析完成,页面中的dom元素已经可用。但是页面中引用的图片、subframe可能还没有加载完。

OnLoad:页面的所有资源都加载完毕(包括图片)。浏览器的载入进度在这时才停止。

这两个时间点将页面加载的timeline分成了三个阶段。

4.异步加载的其它方法

由于Javascript动态特性,还有很多异步加载方法:

· XHR Eval 

· XHR Injection

· Script in Iframe

· Script Defer

· document.write Script Tag

· 还有一种方法是用 setTimeout 延迟0秒 与 其它方法组合。

XHR Eval :通过 ajax 获取js的内容,然后 eval 执行。

var xhrObj = getXHRObject(); 
 xhrObj.onreadystatechange =  
   function() {  
     if ( xhrObj.readyState != 4 ) return; 
     eval(xhrObj.responseText); 
   }; 
 xhrObj.open('GET', 'A.js', true); 
 xhrObj.send('');
 

Script in Iframe:创建并插入一个iframe元素,让其异步执行 js 

var iframe = document.createElement('iframe'); 
 document.body.appendChild(iframe); 
 var doc = iframe.contentWindow.document; 
 doc.open().write('<body onload="insertJS()">'); 
 doc.close();
 

GMail Mobile:页内 js 的内容被注释,所以不会执行,然后在需要的时候,获取script元素中 text 内容,去掉注释后 eval 执行。

<script type="text/javascript"> 
 /* 
 var ...  
 */ 
 </script>
 

详见参考资料中2010年的Velocity 大会 Steve Souders 和淘宝的那两个讲义。

二、async 和 defer 属性

1. defer 属性

<script src="file.js" defer></script> 

defer属性声明这个脚本中将不会有 document.write 或 dom 修改。

浏览器将会并行下载 file.js 和其它有 defer 属性的script,而不会阻塞页面后续处理。

defer属性在IE 4.0中就实现了,超过13年了!Firefox 从 3.5 开始支持defer属性 

注:所有的defer 脚本保证是按顺序依次执行的。 

2. async 属性

<script src="file.js" async></script> 

async属性是HTML5新增的。作用和defer类似,但是它将在下载后尽快执行,不能保证脚本会按顺序执行。它们将在onload 事件之前完成。

Firefox 3.6Opera 10.5IE 9 和 最新的Chrome 和 Safari 都支持 async 属性。可以同时使用 async 和 defer,这样IE 4之后的所有 IE 都支持异步加载。

3. 详细解释

<script> 标签在 HTML 4.01 与 HTML5 的区别:

· type 属性在HTML 4中是必须的,在HTML5中是可选的。

· async 属性是HTML5中新增的。

· 个别属性(xml:space)在HTML5中不支持。

说明:

1. 没有 async 属性,script 将立即获取(下载)并执行,然后才继续后面的处理,这期间阻塞了浏览器的后续处理。

2. 如果有 async 属性,那么 script 将被异步下载并执行,同时浏览器继续后续的处理。

3. HTML4中就有了defer属性,它提示浏览器这个 script 不会产生任何文档元素(没有document.write),因此浏览器会继续后续处理和渲染。

4. 如果没有 async 属性 但是有 defer 属性,那么script 将在页面parse之后执行。

5. 如果同时设置了二者,那么 defer 属性主要是为了让不支持 async 属性的老浏览器按照原来的 defer 方式处理,而不是同步方式。

另参见官方说明:script async 

个人补充:

既然 HTML5 中已经支持异步加载,为什么还要使用前面推荐的那种麻烦(动态创建 script 元素)的方式?

答:为了兼容尚不支持 async 老浏览器。如果将来所有浏览器都支持了,那么直接在script中加上async 属性是最简单的方式。

三、延迟加载(lazy loading

前面解决了异步加载(async loading)问题,再谈谈什么是延迟加载。

延迟加载:有些 js 代码并不是页面初始化的时候就立刻需要的,而稍后的某些情况才需要的。延迟加载就是一开始并不加载这些暂时不用的js,而是在需要的时候或稍后再通过js 的控制来异步加载。

也就是将 js 切分成许多模块,页面初始化时只加载需要立即执行的 js ,然后其它 js 的加载延迟到第一次需要用到的时候再加载。

特别是页面有大量不同的模块组成,很多可能暂时不用或根本就没用到。

就像图片的延迟加载,在图片出现在可视区域内时(在滚动条下拉)才加载显示图片。

四、script 的两阶段加载 与 延迟执行(lazy execution

JS的加载其实是由两阶段组成:下载内容(download bytes)和执行(parse and execute)。

浏览器在下载完 js 的内容后就会立即对其解析和执行,不管是同步加载还是异步加载。

前面说的异步加载,解决的只是下载阶段的问题,但代码在下载后会立即执行。

而浏览器在解析执行 JS 阶段是阻塞任何操作的,这时的浏览器处于无响应状态。

我 们都知道通过网络下载 script 需要明显的时间,但容易忽略了第二阶段,解析和执行也是需要时间的。script的解析和执行所花的时间比我们想象的要多,尤其是script 很多很大的时候。有些是需要立刻执行,而有些则不需要(比如只是在展示某个界面或执行某个操作时才需要)。

这些script 可以延迟执行,先异步下载缓存起来,但不立即执行,而是在第一次需要的时候执行一次。

利用特殊的技巧可以做到 下载 与 执行的分离 (再次感谢 javascript 的动态特性)。比如将 JS 的内容作为 Image或 object 对象加载缓存起来,所以就不会立即执行了,然后在第一次需要的时候再执行。

此部分的更多解释 请查看末尾参考资料中 ControlJS 的相关链接。 

小技巧:

1. 模拟较长的下载时间

写个后端脚本,让其 sleep 一定时间。如在 jsp  Thread.sleep(5000); ,这样5秒后才能收到内容。

2. 模拟较长的 js 代码执行时间(因为这步一般比较快不容易观察到):

var t_start = Number(new Date());

while ( t_start + 5000 > Number(new Date()) ) {}

这个代码将使 js 执行5秒才能完成!

五、script 标签使用的历史

 

1. script 放在 HEAD 

<head>
  <script src=“…”></script>
  </head>

· 阻止了后续的下载;

· IE 6-7 中 script 是顺序下载的,而不是现在的 并行下载、顺序执行” 的方式;

· 在下载解析执行阶段阻止渲染(rendering);

2. script 放在页面底部(2007

... 
 <script src=“…”></script> 
 </body>

· 不阻止其它下载;

· IE 6-7 中 script 是顺序下载的;

· 在下载解析执行阶段阻止渲染(rendering);

3. 异步加载script2009

var se = document.createElement
 ('script'); 
 se.src = 'http://anydomain.com/A.js'; 
 document.getElementsByTagName('head') 
 [0].appendChild(se);

这就是本文主要说的方式。

· 不阻止其它下载;

· 在所有浏览器中,script都是并行下载;

· 只在解析执行阶段阻止渲染(rendering);

4. 异步下载 按需执行 (2010)

var se = new Image(); 
 se.onload = registerScript(); 
 se.src = 'http://anydomain.com/A.js';
 把下载 js 与 解析执行 js 分离出来

· 不阻止其它下载;

· 在所有浏览器中,script都是并行下载;

· 不阻止渲染(rendering)直到真正需要时;

六、异步加载的问题

在异步加载的时候,无法使用 document.write 输出文档内容。

在同步模式下,document.write 是在当前 script 所在的位置输 出文档的。而在异步模式下,浏览器继续处理后续页面内容,根本无法确定 document.write 应该输出到什么位置,所以异步模式下 document.write 不可行。而到了页面已经 onload 之后,再执行 document.write 将导致当前页面的内容被清空,因为它会自动触发 document.open 方法。

实际上document.write的名声并不好,最好少用。

替代方法:

1. 虽然异步加载不能用 document.write,但还是可以onload之后执行操作dom(创建dom或修改dom)的,这样可以实现一些自己的动态输出。比如要在页面异步创建一个浮动元素,这和它在页面中的位置就没关系了,只要创建出该dom元素添加到 document 中即可。

2. 如果需要在固定位置异步生成元素的内容,那么可以在该固定位置设置一个dom元素作为目标,这样就知道位置了,异步加载之后就可以对这个元素进行修改。

六、JS 模块化管理

异步加载,需要将所有 js 内容按模块化的方式来切分组织,其中就存在依赖关系,而异步加载不保证执行顺序。

另外,namespace 如何管理 等相关问题。这部分已超出本文内容,可参考:

RequireJS  CommonJS 以及 王保平(淘宝) SeaJS 及其博客 

七、JS最佳实践:

1. 最小化 js 文件,利用压缩工具将其最小化,同时开启http gzip压缩。工具:

2. 尽量不要放在 <head> 中,尽量放在页面底部,最好是</body>之前的位置

3. 避免使用 document.write 方法

4. 异步加载 js ,使用非阻塞方式,就是此文内容。

5. 尽量不直接在页面元素上使用 Inline Javascript,如onClick 。有利于统一维护和缓存处理。

参考资料:

<!-- 只在首页引入 --> 

 

<script type="text/javascript" src="/js/touchslider.js?ver=20122101"></script> 

   
 

<script type="text/javascript"> 

 

/* 登陆配置 */ 

 

window.LOGIN_CFG=window.LOGIN_CFG||{ 

 

blank:'http://uc.duopao.com/uc/blank.html?ver=20122101', 

 

loginURL:'http://uc.duopao.com/uc/hlogin' 

 

 

</script> 

 

<!-- end 只在首页引入 --> 

 

<script type="text/javascript" src="/js/common.js?ver=20122101"></script> 

 

<script type="text/javascript"> 

 

var _gaq = _gaq || []; 

 

_gaq.push(['_setAccount', 'UA-25424360-1']); 

 

_gaq.push(['_setDomainName', '.duopao.com']); 

 

_gaq.push(['_addOrganic', 'soso', 'w']); 

 

_gaq.push(['_addOrganic', 'yodao', 'q']); 

 

_gaq.push(['_addOrganic', 'sogou', 'query']); 

 

_gaq.push(['_addOrganic', 'so.360.cn', 'q']); 

 

_gaq.push(['_trackPageview']); 

 

_gaq.push(['_trackPageLoadTime']); 

   
 

(function() { 

 

var ga = document.createElement('script'); ga.type = 'text/javascript'; ga.async = true; 

 

ga.src = ('https:' == document.location.protocol ? 'https://ssl' : 'http://www') + '.google-analytics.com/ga.js'; 

 

var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(ga, s); 

 

})(); 

   
 

</script> 

 

 

 

 

<script type="text/javascript"> 

 

var _bdhmProtocol = (("https:" == document.location.protocol) ? " https://" : " http://"); 

 

document.write(unescape("%3Cscript src='" + _bdhmProtocol + "hm.baidu.com/h.js%3F0bde07579b8fa55c37b5a5d2540cb781' type='text/javascript'%3E%3C/script%3E")); 

 

</script> 

注释:1首先判断当前的协议是什么如果是https_bdhmProtocol="https://"否则_bdhmProtocol="http://"
然后解码这个"%3Cscript src='" + _bdhmProtocol + "hm.baidu.com/h.js%3F0ea9650be45bfb0dc6d786406ad08916' type='text/javascript'%3E%3C/script%3E"
经过escape编码的字符串

2首先是用var声明一个变量 变量名称是_bdhmProtocol(网络通信协议),然后就是后面的全部都是这个变量的值;
      括号里面的第一个小括号是一个三木运算符,意思是 “http:" 等于document.location.protocol吗?等于就执行前面的值,不等于就执行后面的的那个值, 这第一个小括号里就是字符串 " https:// “或者 " http:// " 
    document.write()//屏幕输出知道吧 
    unescape 返回指定值的 ASCII 字符串(ECMAScript v3 已从标准中删除)详情网上可查阅
   输出的是一个指向的网址 

3在网页相应的地方加上以上代码,就会显示一个baidu的查询框出来。

 

2
1
分享到:
评论

相关推荐

    复古怀旧教室桌椅素材同学聚会毕业纪念册模板.pptx

    复古怀旧教室桌椅素材同学聚会毕业纪念册模板

    数据结构与算法中的回文串:马拉车算法与中心扩展法的Python实现及其应用

    内容概要:本文详细介绍了关于寻找最长回文子串的相关知识,包括前置理论、不同求解方法的具体操作及其优势。首先,阐述了回文串这一特殊结构的概念及其特性——正序逆序完全一致。紧接着,围绕中心扩展法和马拉车算法展开叙述。前者基于字符逐一检查周边字符的方法,在每一点向外延伸探索最长相同序列。而后者利用预处理阶段将字符间隔用特殊符号填充形成全新字符串形式,并建立相应的辅助工具(如回文半径数组)用于跟踪回文中点位置与范围变动情况,再经由巧妙规则判定并逐步推进搜索进度,从而大大降低了运行成本。最后还提到了两种常规方法——动态规划与中心扩展的具体思路以及它们各自适用的情况和局限性。 适合人群:正在研究或学习数据结构和算法特别是字符串处理相关领域的技术人员和爱好者。 使用场景及目标:帮助开发者针对字符串匹配类的问题,尤其是回文判定和查找场景,掌握更多高效的解决方案,优化程序性能。具体应用场景如文本编辑器、搜索引擎中的关键字检索等方面可能会涉及到这类算法的应用。

    前端分析-2023071100789

    前端分析-2023071100789

    国货彩妆品牌Q1社媒营销投放分析报告.pdf

    国货彩妆品牌Q1社媒营销投放分析报告

    内螺旋覆盖算法(ISC)matlab代码

    内螺旋覆盖算法(ISC)是一种用于填充正方形矩阵的算法。它的主要思想是按照一定的规则将矩阵从外部向内部进行覆盖,直到矩阵的中心被填充。在覆盖的环境区域内,机器人按一定的方向如顺时针或逆时针进行覆盖,当前方有未覆盖的栅格时机器人就向前运动,如果前方有障碍物或者已经被覆盖过,则机器要向右(或向左)旋转 90 °继续向前行走。

    21电平MMC整流站、MMC逆变站、两端柔性互联的MATLAB仿真模型,4端柔性互联、MMC桥臂平均值模型、MMC聚合模型(四端21电平一分钟即能完成2s的工况仿真) 1-全部能正常运行,图四和图五为

    21电平MMC整流站、MMC逆变站、两端柔性互联的MATLAB仿真模型,4端柔性互联、MMC桥臂平均值模型、MMC聚合模型(四端21电平一分钟即能完成2s的工况仿真) 1-全部能正常运行,图四和图五为仿真波形 2-双闭环控制,逆变站PQ控制,整流站站Udc Q控制 3-最近电平逼近调制+子模块电容充电 4-环流抑制控制

    永磁同步电机电流滞环控制Matlab simulink仿真模型,参数已设置好,可直接运行 属于PMSM转速电流双闭环矢量控制系统模型 电流内环采用电流滞环控制(pang-pang控制),转速外环为

    永磁同步电机电流滞环控制Matlab simulink仿真模型,参数已设置好,可直接运行。 属于PMSM转速电流双闭环矢量控制系统模型。 电流内环采用电流滞环控制(pang-pang控制),转速外环为PI控制。 波形完美,包含原理说明文档和参考文献。

    基于java的项目申报管理系统设计与实现.docx

    基于java的项目申报管理系统设计与实现.docx

    Spring Boot框架下的权限管理与工作流开发平台系统实现

    项目介绍 Spring Boot + Security + MyBatis Plus 快速开发平台 内置功能 用户管理:用户是系统操作者,该功能主要完成系统用户配置。 权限管理:配置系统菜单,操作权限,按钮权限, 数据权限标识等。 角色管理:角色菜单权限分配、设置角色按机构进行数据范围权限划分。 字典管理:对系统中经常使用的一些较为固定的数据进行维护。 参数管理:对系统动态配置常用参数。 通知公告:系统通知公告信息发布维护。 操作日志:系统正常操作日志记录和查询;系统异常信息日志记录和查询。 登录日志:系统登录日志记录查询包含登录异常。 定时任务:在线(添加、修改、删除)任务调度包含执行结果日志。 代码生成:前后端代码的生成(java、html、xml、sql)支持CRUD下载 。 系统接口:根据业务代码自动生成相关的api接口文档。 服务监控:监视当前系统CPU、内存、磁盘、堆栈等相关信息。 表单构建:拖动表单元素生成相应的HTML代码。 数据监视:监视当前系统数据库连接池状态,可进行分析SQL找出系统性能瓶颈。 租户管理:加入多租户架构, 使用逻辑隔离租户数据。

    3.0版-聊天业务实现.7z

    3.0版-聊天业务实现.7z

    Hyperledger Fabric支持的分布式身份可信认证区块链解决方案(含链码、启动脚本及SDK).zip

    Hyperledger Fabric支持的分布式身份可信认证区块链解决方案(含链码、启动脚本及SDK).zip [资源说明] 1、该项目是团队成员近期最新开发,代码完整,资料齐全,含设计文档等 2、上传的项目源码经过严格测试,功能完善且能正常运行,请放心下载使用! 3、本项目适合计算机相关专业(人工智能、通信工程、自动化、电子信息、物联网等)的高校学生、教师、科研工作者、行业从业者下载使用,可借鉴学习,也可直接作为毕业设计、课程设计、作业、项目初期立项演示等,也适合小白学习进阶,遇到问题不懂就问,欢迎交流。 4、如果基础还行,可以在此代码基础上进行修改,以实现其他功能,也可直接用于毕设、课设、作业等。 5、不懂配置和运行,可远程教学 欢迎下载,学习使用!

    ffmpeg-7.1安装包.zip

    ffmpeg 是一个非常强大且灵活的多媒体处理工具,广泛用于视频和音频的转码、剪辑、转换格式、调整尺寸、添加水印等多种操作。

    【锂电池剩余寿命预测】CNN卷积神经网络锂电池剩余寿命预测,马里兰大学锂电池数据集(Pytorch完整源码和数据)

    1.【锂电池剩余寿命预测】CNN卷积神经网络锂电池剩余寿命预测,马里兰大学锂电池数据集(Pytorch完整源码和数据) 2.数据集:马里兰大学锂电池数据集,已经处理好; 3.环境准备:python 3.8 , pytorch 1.8 版本及其以上,代码格式ipynb文件,可读性强; 4.模型描述:CNN-Transformer在各种各样的问题上表现非常出色,现在被广泛使用。 5.领域描述:近年来,随着锂离子电池的能量密度、功率密度逐渐提升,其安全性能与剩余使用寿命预测变得愈发重要。本代码实现了CNN卷积神经网络在该领域的应用。 6.作者介绍:机器学习之心,博客专家认证,机器学习领域创作者,2023博客之星TOP50,文章底部有博主联系方式。从事Matlab、Python算法仿真工作8年,更多仿真源码、数据集定制私信。

Global site tag (gtag.js) - Google Analytics