异步载入JS

  平时最常使用的就是这样的同步载入形式:
   <script src="http://yourdomain.com/script.js"></script>
   同步模式。又称堵塞模式,会阻止浏览器的兴许处理,停止了兴许的解析。因此停止了兴许的文件载入(如图像)、渲染、代码运行。


js 之所以要同步运行。是由于 js 中可能有输出 document 内容、改动dom、重定向等行为,所以默认同步运行才是安全的。
   曾经的一般建议是把<script>放在页面末尾</body>之前,这样尽可能降低这样的堵塞行为,而先让页面展示出来。


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

常见异步载入(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); 
})(); 


比如 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); 
})(); 

百度地图的异步载入:

	var map;  //全局地图
	function initialize() {
		map = new BMap.Map('map');
		var point = new BMap.Point(121.491, 31.233);
		map.centerAndZoom(point, 11);
		//使用鼠标滚轮控制缩放
		map.enableScrollWheelZoom();
		//加入地图控件
		map.addControl(new BMap.NavigationControl());
		map.addControl(new BMap.ScaleControl());
		map.addControl(new BMap.OverviewMapControl());
		map.addControl(new BMap.MapTypeControl());
		//点击地图就加入标注点
		map.addEventListener("click", addMarkerService);
		PanOptions.noAnimation=false;
	}
	function loadScript() {
		var script = document.createElement("script");
		script.src = "http://api.map.baidu.com/api?

v=1.5&ak={No}&callback=initialize"; //此为v1.5版本号的引用方式 // http://api.map.baidu.com/api?

v=1.5&ak=您的密钥&callback=initialize"; //此为v1.4版本号及曾经版本号的引用方式 document.body.appendChild(script); } window.onload = loadScript;


可是。这样的载入方式在载入运行完之前会阻止 onload 事件的触发,而如今非常多页面的代码都在 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分成了三个阶段。

 

异步载入的其他方法
由于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(''); 
posted @ 2018-01-20 16:20  llguanli  阅读(156)  评论(0编辑  收藏  举报