最近因为工作需要做了一个js自动导入的插件,一开始很天真的以为动态创建个script添加到head中就ok了,试了之后才发现了问题,就是如果同时引入了多个js文件,而且后一个文件中用到了前一个文件中的变量,那就会报错,靠~~悲催了,就是说js如果动态加载(非浏览器加载因为浏览器加载时同步加载的会等待前一个js加载完成后才进行下一个js加载,这样就不会出现问题)那加载的文件是异步进行的,难怪啊!然后在网上找了些资料说用ajax同步加载,然后我试了真可以,下面就是我的代码分享出来给大家,但是注意这样加载出来的js有一个致命的弱点,就是没法再浏览器上调试- - !!!!!以后再寻找解决方案吧,先贴代码:
js主文件importComJs.js:
**
* Created by carlos on 2015/5/19.
* QImport 导入帮助函数,可以很方便的导入指定的js、css文件。但是需要注意的是,这样导入的js将无法调试,暂时没有找到解决方案--!
* 使用方法先把此文件引入,必须放在head中的最前面导入,
* <script src="../js/importComJs.js" id="QImport"></script>
* id必须是QImport,否则无效
* 然后加入
* <script>
* QImport.init(customscripts);
* </script>
* customscripts参数为:外加的引入文件配置,可以不填。但填写必须遵守QImport.IMPORTSCRIPTS的写法。
* 如果不想手动初始化引入函数,可以这样写:
* <script src="../js/importComJs.js" id="QImport" data-auto="true"></script>
* 这里的 data-auto为true时候才会自动初始化,否则需要手动初始化。
* 使用上自动初始化,不用担心引入顺序问题,后面的其他js、css文件按正常引用就可以了。
* 注意:importComjs文件必须第一个引入,否则将出现引用找不到的问题。
* 新增属性:
* data-config:自定义配置文件名,默认为“importConfig.json”
* data-personalconfigname:自定义个性化配置名称,在配置文件中的“personalscripts”中配置
* 注意:配置加载顺序是:importscripts最高、personalscripts其次、customscripts最低。如果引入的次序不对会引起空指针报错哦^^!
*/ /**
* 同步加载js脚本
* @param id 需要设置的<script>标签的id
* @param url js文件的相对路径或绝对路径
* @return {Boolean} 返回是否加载成功,true代表成功,false代表失败
*/
(function(){
QImport = {
self:{
obj:function(){
return document.getElementById('QImport'); //获取script的id,如果项目中实在是有冲突不得不改,那就该这个“QImport”吧!
},
auto:function(){
return eval(this.getAttr('data-auto')); //是否开启自动初始化模式,默认为false
},
config:function(){
return this.getAttr('data-config'); //自定义配置文件名,默认为“importConfig.json”
},
personal:function(){
return this.getAttr('data-personalconfigname') //自定义个性化配置名称,在配置文件中的“personalscripts”中配置。
},
src:function(){
return this.getAttr('src'); //此文件位置,不多说
},
getAttr:function(attrName){
var attrValue = undefined;
try{
attrValue = this.obj().getAttribute(attrName);
}
catch(e){}
return attrValue;
},
relationpath:function(){
var src = this.src();
var re = /^[..\/]*/;
return src.match(re)[0];
},
path:function(filename){
var src = this.src();
var temp = src.split('/');
var path = "";
temp.pop();
temp.push(filename);
return temp.join('/');
}
},
autoinit:function(){
if(this.self.auto()){
this.init();
}
},
getpath:function(p){
var path = this.self.relationpath()+p;
return path.replace('//','/');
},
init:function(customscripts){ //导入文件的主方法
this.extend(customscripts);
var IMPORTSCRIPTS = this.IMPORTSCRIPTS;
var head = document.getElementsByTagName("head").item(0);
var meta = document.createElement('meta'); //自动添加web手机适应代码;
meta.name = 'viewport';
meta.content = 'width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no';
head.insertBefore(meta, head.childNodes[0]);
var re = /^http[s]?:\/\//i;
for(var i=0;i<IMPORTSCRIPTS.length;i++){
loadpath = IMPORTSCRIPTS[i].url;
if(!re.test(IMPORTSCRIPTS[i].url)){
loadpath = this.getpath(IMPORTSCRIPTS[i].url);
}
if(IMPORTSCRIPTS[i].type=="script"){
var flag = this.loadJS(loadpath,loadpath,i+1)
if(!flag){
alert('loading path:"'+loadpath+'" failed,May be lost "http(s)://"');
}
}
else if(IMPORTSCRIPTS[i].type=="css"){
var csss = document.createElement('link')
csss.href = this.getpath(IMPORTSCRIPTS[i].url);
csss.rel = 'stylesheet';
head.insertBefore(csss, head.childNodes[i+1]);
}
}
},
extend:function(customscripts){ //自定义扩展方法,仅适用当前对象
if(!(customscripts instanceof Array)) return;
var IMPORTSCRIPTS = this.IMPORTSCRIPTS;
for(var i=0;i<customscripts.length;i++){
var flag = false;
for(var j=0;j<IMPORTSCRIPTS.length;j++){
if((IMPORTSCRIPTS[j].type==customscripts[i].type)&&(IMPORTSCRIPTS[j].url==customscripts[i].url)){
flag = true;
break;
}
}
if(!flag){
this.IMPORTSCRIPTS.push({type:customscripts[i].type,url:customscripts[i].url});
}
}
},
ajax:function(url){ //ajax原始方法,这里仅可以同步请求
var xmlHttp = null;
if(window.ActiveXObject)//IE
{
try {
//IE6以及以后版本中可以使用
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP");
}
catch (e) {
//IE5.5以及以后版本可以使用
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
}
}
else if(window.XMLHttpRequest)//Firefox,Opera 8.0+,Safari,Chrome
{
xmlHttp = new XMLHttpRequest();
}
//采用同步加载
xmlHttp.open("GET",url,false);
//发送同步请求,如果浏览器为Chrome或Opera,必须发布后才能运行,不然会报错
xmlHttp.send(null);
//4代表数据发送完毕
if ( xmlHttp.readyState == 4 ){
//0为访问的本地,200到300代表访问服务器成功,304代表没做修改访问的是缓存
if((xmlHttp.status >= 200 && xmlHttp.status <300) || xmlHttp.status == 0 || xmlHttp.status == 304){
return xmlHttp.responseText;
}
else{
return false;
}
}
else{
return false;
}
},
loadJS:function(id,url,rank){ //加载js文件
var importText = this.ajax(url);
if(!importText) return false;
var myHead = document.getElementsByTagName("HEAD").item(0);
var myScript = document.createElement( "script" );
myScript.language = "javascript";
myScript.type = "text/javascript";
myScript.id = id;
try{
//IE8以及以下不支持这种方式,需要通过text属性来设置
myScript.appendChild(document.createTextNode(importText));
}
catch (ex){
myScript.text = importText;
}
myHead.insertBefore(myScript, myHead.childNodes[rank]);
return true;
},
template:{ //加载模板文件,现在仅在引入“jquery”的情况下可用。
parent:this, //父类this引用
_setparent:function(){ //把父类的this赋给子方法的parent
this.template.parent = this;
},
includeTagName:'include', //引入时候用的标签:<include url="head.html"></include>
init:function(){
this.include();
},
include:function(){
var includes = document.getElementsByTagName(this.includeTagName);
for(var i=0;i<includes.length;i++){
var url = includes[i].getAttribute('url');
var includeHtml = this.parent.ajax(url);
var parent = includes[i].parentNode;
var newEs = this.parseDom(includeHtml);
for(var j=0;j<newEs.length;j++){
parent.insertBefore(newEs[j], includes[i])
}
}
while(true){
if(includes.length==0) break;
includes[0].remove();
}
},
parseDom:function(str){
var obj = document.createElement('div');
obj.innerHTML = str;
return obj.childNodes;
}
},
initparent:function(){ //子类中想用父类中的方法必须通过此方法初始化以下,还必须引入如template的parent和_setparent
this.template._setparent.call(this);
},
getimportjson:function(){ //获取的配置json
var configName = this.self.config(); //共用配置
if(configName) this.configname = configName;
var configpath = this.self.path(this.configname);
var confJsonStr = this.ajax(configpath);
if(!confJsonStr){
alert('config file path:"'+configpath+'" can not be loaded');
return;
}
this.confJson = eval('('+confJsonStr+')');
importarr = this.confJson.importscripts;
if(importarr instanceof Array){
this.IMPORTSCRIPTS = this.confJson.importscripts;
}
this.getpersonalarr();
},
getpersonalarr:function(){ //获取个性化配置arr
var personalConfigName = this.self.personal();
if(!personalConfigName) return;
this.personalconfigname = personalConfigName;
var personalarr = this.confJson.personalscripts[personalConfigName];
if(personalarr instanceof Array){
this.extend(personalarr)
}
},
confJson:{},
IMPORTSCRIPTS:[],
configname:'importConfig.json'
}
QImport.getimportjson();
QImport.autoinit();
QImport.initparent();
try{
$(function(){
QImport.template.init();
});
}
catch(e){} })();
下面是配置文件,importConfig.json:
{
importscripts:[
{
type: 'script',
url: 'js/jquery-1.9.1/jquery-1.9.1.min.js'
},{
type: 'script',
url: 'js/mui.min.js'
},{
type: 'css',
url: 'css/mui.min.css'
},{
type: 'script',
url: 'js/json2.js'
}
],
personalscripts:{
test:[
{
type:'script',
url:'js/test.js'
}
]
}
}
注意配置文件必须和导入方法js在同一个路径底下。
以上就是我个人封装的引入js、css的代码,希望和大家交流下,有什么问题希望大家能不吝指教!