标签:
一种常见的做法就是将<script>标签放在闭合的</body>之前。这样就可以先把页面展示给用户,让页面的加载速度显得不是很慢。这时最好将CSS文件放在head中,一边加载DOM一边渲染样式。
我们知道多个HTTP请求也会降低页面性能。因此我们可以采用将多个脚本文件合并到一个文件下载。这虽然减少了HTTP请求,但这给我们增加了较大的额外工作,因为每次发布我们都要合并文件。为此,我们可以采用成组下载的方式来达到目的。成组下载就是一次请求下载多个脚本文件。例如以下的URL:每次向服务器的固定服务请求下载多个文件。服务器将多个文件合并一起返回给客户端。这是HTML页面包含多个外部JavaScript的最佳方法。
如果script文件又大又多,那么恐怕我们怎么压缩文件怎么减少http请求数,script的加载都会锁定浏览器一大段时间。这时比较好的方法就是等页面加载完成后再加载script,也就是在window.onload事件发出之后开始下载代码。HTML4中的defer属性可以实现下载script时不阻塞浏览器其他处理过程,代码下载完成后也不会被执行,而是等到DOM加载完成后(window.onload之前)才被执行。但defer属性只被IE4+和FF3.5+支持,其他浏览器不支持该属性,因此这不是一个理想的跨浏览器解决方案。延时加载的一种通用方法就是动态创建脚本元素。一个新的<script>元素可以非常容易地通过标准DOM函数创建:var script = document.createElement("script");script.type = "text/javascript";script.src = "xxx.js";document.getElementsByTagName("head")[0].appendChild(script);当script元素被添加到页面之后便开始下载脚本文件。该技术的优点是:无论在何处启动下载,文件的下载和运行都不会阻塞其他页面处理过程。当script文件下载完成后,返回的代码通常被立即执行(除了FF和Opera,它们将等待此前的所有动态脚本节点执行完毕。)
function loadScript(url, callback){
var script = document.createElement("script");
script.type = "text/javascript";
if(script.readyState){ // IE
script.onreadystatechange = function(){
if(script.readyState == "loaded" || script.readyState == "complete"){
script.onreadystatechange = null;
callback();
}
};
}else{ // FF, Chrome, Opera, ...
script.onload = function(){
callback();
};
}
script.src = url;
document.getElementsByTagName("head")[0].appendChild(script);
}
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>动态加载</title>
<script type="text/javascript" src="../scripts/loadScript.js"></script>
<script type="text/javascript">
loadScript("../scripts/test1.js", function(){
loadScript("../scripts/test2.js", function(){
loadScript("../scripts/test3.js", function(){
test3.print("hi, i‘m abc.");
})
});
});
</script>
</head>
<body>
</body>
用于测试的test1.js:var test2 = {print: function(msg){test1.print(msg);alert("from test2: " + msg);}};
var test3 = {print: function(msg){test2.print(msg);alert("from test3: " + msg);}};
/**
* create xmlhelper
* */
(function(window){
var xhrhelper = (function(){
var xhrhelper = function(){};
xhrhelper.extend = function(){
var len = arguments.length, target = this;
for(var i = 0; i < len; i++){
var source = arguments[i];
for(var p in source){
if(target[p] == undefined){
target[p] = source[p];
}else{
throw new Error("extend error: "+ p +" is already existed.");
}
}
}
};
return xhrhelper;
})();
window.xhrhelper = xhrhelper;
})(window);
/**
* extend xmlhelper
* */
xhrhelper.extend({
/**
* create XmlHttpRequest object.
* */
createXmlHttp: function(){
var xmlhttp;
/*@cc_on
@if (@_jscript_version >= 5)
try{
xmlhttp = new ActiveXObject("Msxml2.XMLHTTP");
} catch(e){
try{
xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");
} catch(E){
xmlhttp = false;
}
}
@else
xmlhttp = false;
@end
@*/
if(!xmlhttp && typeof XMLHttpRequest != ‘undefined‘){
try{
xmlhttp = new XMLHttpRequest();
} catch(e){
xmlhttp = false;
}
}
return xmlhttp;
},
/**
* send request.
* */
req: function(options){
var defaults = {
method: "get", // GET,POST,PUT,DELTE
url: null,
data: null,
async: true,
username: null,
password: null,
onloading: function(xhr){},
onloaded: function(xhr){},
oninteractive: function(xhr){},
oncomplete: function(xhr){},
onabort: function(xhr){},
onerror: function(xhr){}
};
for(var p in options){
defaults[p] = options[p];
}
var xmlhttp = xhrhelper.createXmlHttp();
if(xmlhttp){
xmlhttp.onreadystatechange = function(){
if(xmlhttp.readyState == 1){ // 未初始化
defaults.onloading(xmlhttp);
} else if(xmlhttp.readyState == 2){ // 已初始化
defaults.onloaded(xmlhttp);
} else if(xmlhttp.readyState == 3){
defaults.oninteractive(xmlhttp);
} else if(xmlhttp.readyState == 4){
if(xmlhttp.status == 0){
defaults.onabort(xmlhttp);
} else if(xmlhttp.status >= 200 && xmlhttp.status <= 300 || xmlhttp.status == 304){
defaults.oncomplete(xmlhttp);
} else{
defaults.onerror(xmlhttp);
}
}
};
if(defaults.username != null && defaults.password != null &&
defaults.username.length != 0 && defaults.password.length != 0){
xmlhttp.open(defaults.method, defaults.url, defaults.async,
defaults.username, defaults.password);
}else{
xmlhttp.open(defaults.method, defaults.url, defaults.async);
}
if(defaults.method.toLowerCase() == "get"){
}else if(defaults.method.toLowerCase() == "post"){
xmlhttp.setRequestHeader(‘Content-Type‘, ‘text/html; charset=UTF-8‘);
}
xmlhttp.send(defaults.data);
}
return xmlhttp;
}
});
然后是利用xhr下载js文件的函数:function xhrload(url, callback){
xhrhelper.req({
method: "get",
url: url,
oncomplete: function(xhr){
var script = document.createElement("script");
script.type = "text/javascript";
script.text = xhr.responseText;
document.body.appendChild(script);
callback();
}
});
}
在页面中可以通过如下方式调用xhrload函数:xhrload("../scripts/test1.js", function(){test1.print("ABC");});
(转)高性能JavaScript:加载和运行(动态加载JS代码)
标签:
原文地址:http://www.cnblogs.com/susufufu/p/5764431.html