实例讲解避免javascript冲突的方法


Posted in Javascript onJanuary 03, 2016

本文实例讲解了如何避免javascript中冲突的方法,需要的朋友可以了解一下

[1]工程师甲编写功能A

var a = 1;
var b = 2;
alert(a+b);//3

[2]工程师乙添加新功能B

var a = 2;
var b = 1;
alert(a-b);//1

[3]上一步中,工程师乙在不知情的情况下,定义了同名变量a,产生冲突。于是使用匿名函数将脚本包起来,让变量作用域控制在匿名函数之内。

//功能A
(function(){
var a = 1;

var b = 2;

alert(a+b);//3
})();
//功能B
(function(){

var a = 2;

var b = 1;

alert(a-b);//1
})();

[4]此时有了新需求,网页中加入功能C,且需要用到功能A中的变量b。于是在window作用域下定义一个全局变量,把它作为一个桥梁,完成各匿名函数之间的通信

//全局变量
var str;
//功能A
(function(){
var a = 1;

//将b的值赋给str

var b = str = 2;

alert(a+b);//3
})();
//功能B
(function(){

var a = 2;

var b = 1;

alert(a-b);//1
})();
//功能C
(function(){

//将str的值赋给b

var b = str;

alert(b);//2
})();

[5]但如果功能C还需要功能A中的变量a呢,这时就需要再定义一个全局变量

//全局变量
var str,str1;
//功能A
(function(){
//将a的值赋给str1

var a = str1 = 1;

//将b的值赋给str

var b = str = 2;

alert(a+b);//3
})();
//功能B
(function(){

var a = 2;

var b = 1;

alert(a-b);//1
})();
//功能C
(function(){

//将str1的值赋给a

var a = str1;

//将str的值赋给b

var b = str;

alert(a*b);//2
})();

[6]但随着匿名函数之间需要通信的变量越多,需要的全局变量也就越多。因此需要严格控制全局变量的数量,使用hash对象作为全局变量,可以将需要的变量都作为对象的属性,可以保证全局变量的个数足够少,同时拓展性非常好

//全局变量
var GLOBAL = {};
//功能A
(function(){
//将a的值赋给GLOBAL.str1

var a = GLOBAL.str1 = 1;

//将b的值赋给GLOBAL.str

var b = GLOBAL.str = 2;

alert(a+b);//3
})();
//功能B
(function(){

var a = 2;

var b = 1;

alert(a-b);//1
})();
//功能C
(function(){

//将GLOBAL.str1的值赋给a

var a = GLOBAL.str1;

//将GLOBAL.str的值赋给b

var b = GLOBAL.str;

alert(a*b);//2
})();

[7]但如果新增功能D,功能D需要和功能B通信,并使用功能B脚本中的变量a,开发功能D的是工程师丁

//全局变量
var GLOBAL = {};
//功能A
(function(){
//将a的值赋给GLOBAL.str1

var a = GLOBAL.str1 = 1;

//将b的值赋给GLOBAL.str

var b = GLOBAL.str = 2;

alert(a+b);//3
})();
//功能B
(function(){

//将a的值赋给GLOBAL.str1

var a = GLOBAL.str1 = 2;

var b = 1;

alert(a-b);//1
})();
//功能C
(function(){

//将GLOBAL.str1的值赋给a

var a = GLOBAL.str1;

//将GLOBAL.str的值赋给b

var b = GLOBAL.str;

alert(a*b);//2
})();
//功能D
(function(){

//将GLOBAL.str1的值赋给a

var a = GLOBAL.str1;

alert(a*2);//4
})();

[8]由于工程师丁只关心自己的匿名函数和功能B的匿名函数,使用GLOBAL.str却无意中覆盖了功能A中设置的同名变量,导致功能C出错。于是使用命名空间来解决这个问题,在不同的匿名函数下,根据功能声明一个不同的命名空间,然后每个匿名函数中的GLOBAL对象的属性都不要直接挂在GLOBAL对象上,而是挂在此匿名函数的命名空间下

//全局变量
var GLOBAL = {};
//功能A
(function(){
GLOBAL.A = {};

//将a的值赋给GLOBAL.A.str1

var a = GLOBAL.A.str1 = 1;

//将b的值赋给GLOBAL.A.str

var b = GLOBAL.A.str = 2;

alert(a+b);//3
})();
//功能B
(function(){

GLOBAL.B = {};

//将a的值赋给GLOBAL.B.str1

var a = GLOBAL.B.str1 = 2;

var b = 1;

alert(a-b);//1
})();
//功能C
(function(){

//将GLOBAL.A.str1的值赋给a

var a = GLOBAL.A.str1;

//将GLOBAL.A.str的值赋给b

var b = GLOBAL.A.str;

alert(a*b);//2
})();
//功能D
(function(){

//将GLOBAL.B.str1的值赋给a

var a = GLOBAL.B.str1;

alert(a*2);//4
})();

[9]如果同一个匿名函数中的程序非常复杂,变量名很多,命名空间还可以进一步拓展,生成二级命名空间

//以功能A为例
(function(){
var a = 1, b = 2;

GLOBAL.A = {};

GLOBAL.A.CAT = {};

GLOBAL.A.DOG = {};

GLOBAL.A.CAT.name = 'mimi';

GLOBAL.A.DOG.name = 'xiaobai';

GLOBAL.A.CAT.move = function(){};

GLOBAL.A.str1 = a;

GLOBAL.B.str = b;  
})();

[10]因为生成命名空间是个非常常用的功能,进一步将生成命名空间的功能定义成一个函数,方便调用,完整版本改写后的代码如下

var GLOBAL = {};
GLOBAL.namespace = function(str){
var arr = str.split('.');

var o = GLOBAL;

var start = 0;

if(arr[0] == 'GLOBAL'){


start = 1;

}else{


start = 0;

}

for(var i = start; i < arr.length; i++){


o[arr[i]] = o[arr[i]] || {};


o = o[arr[i]];

}
};
//功能A
(function(){

var a = 1;

var b = 2;

GLOBAL.namespace('A.CAT');

GLOBAL.namespace('A.DOG');

GLOBAL.A.CAT.name = 'mimi';

GLOBAL.A.DOG.name = 'xiaobai';

GLOBAL.A.CAT.move = function(){};

GLOBAL.A.str1 = a;

GLOBAL.A.str = b;  

alert(a+b);//3
})();
//功能B
(function(){

var a = 2;

var b = 1;

GLOBAL.namespace('B');

GLOBAL.B.str1 = a;

alert(a-b);//1
})();
//功能C
(function(){

var a = GLOBAL.A.str1;

var b = GLOBAL.A.str;

alert(a*b);//2
})();
//功能D
(function(){

var a = GLOBAL.B.str1;

alert(a*2);//4
})();

[11]代码的冲突问题已经解决了,但可维护性并不强。比如,现在需要让工程师甲去修改功能B。因为工程师甲写的脚本是关于功能A的,他并不知道功能B的脚本情况。为了改善这种局面,需要给代码添加适当的注释。

var GLOBAL = {};
GLOBAL.namespace = function(str){
var arr = str.split('.');

var o = GLOBAL;

var start = 0;

if(arr[0] == 'GLOBAL'){


start = 1;

}else{


start = 0;

}

for(var i = start; i < arr.length; i++){


o[arr[i]] = o[arr[i]] || {};


o = o[arr[i]];

}
};
/*
* @method 功能A:实现加法运算
* @author 工程师甲
* @connect 1234567
* @time 2015-01-01
*/

(function(){

var a = 1;

var b = 2;

GLOBAL.namespace('A.CAT');

GLOBAL.namespace('A.DOG');

GLOBAL.A.CAT.name = 'mimi';

GLOBAL.A.DOG.name = 'xiaobai';

GLOBAL.A.CAT.move = function(){};

GLOBAL.A.str1 = a;

GLOBAL.A.str = b;  

alert(a+b);//3
})();
/*
* @method 功能B:实现减法运算
* @author 工程师乙
* @connect 1234567
* @time 2015-01-01
*/
(function(){

var a = 2;

var b = 1;

GLOBAL.namespace('B');

GLOBAL.B.str1 = a;

alert(a-b);//1
})();
/*
* @method 功能C:实现乘法运算
* @author 工程师丙
* @connect 1234567
* @time 2015-01-01
*/
(function(){

var a = GLOBAL.A.str1;

var b = GLOBAL.A.str;

alert(a*b);//2
})();
/*
* @method 功能D:实现乘2运算
* @author 工程师丁
* @connect 1234567
* @time 2015-01-01
*/
(function(){

var a = GLOBAL.B.str1;

alert(a*2);//4
})();

让javascript不再冲突,需要

  •   [1]避免全局变量的泛滥
  • [2]合理使用命名空间

  • [3]为代码添加必要的注释

以上就是本文的详细内容,希望对大家的学习有所帮助。

Javascript 相关文章推荐
Js动态添加复选框Checkbox的实例方法
Apr 08 Javascript
javascript生成随机数方法汇总
Nov 12 Javascript
jquery.cookie.js实现用户登录保存密码功能的方法
Apr 15 Javascript
AngularJS操作键值对象类似java的hashmap(填坑小结)
Nov 12 Javascript
详解VueJs异步动态加载块
Mar 09 Javascript
Angular实现图片裁剪工具ngImgCrop实践
Aug 17 Javascript
vue实现将一个数组内的相同数据进行合并
Nov 07 Javascript
Vue项目前后端联调(使用proxyTable实现跨域方式)
Jul 18 Javascript
微信小程序基于ColorUI构建皮皮虾短视频去水印组件
Nov 04 Javascript
JavaScript中Object、map、weakmap的区别分析
Dec 15 Javascript
原生js实现放大镜组件
Jan 22 Javascript
JS实现扫雷项目总结
May 19 Javascript
详解js中class的多种函数封装方法
Jan 03 #Javascript
js中利用tagname和id获取元素的方法
Jan 03 #Javascript
信息页文内画中画广告js实现代码(文中加载广告方式)
Jan 03 #Javascript
基于javascript实现简单计算器功能
Jan 03 #Javascript
详解Javascript事件驱动编程
Jan 03 #Javascript
基于javascript实现仿百度输入框自动匹配功能
Jan 03 #Javascript
js实现搜索框关键字智能匹配代码
Mar 26 #Javascript
You might like
php快速url重写 更新版[需php 5.30以上]
2010/04/20 PHP
PHP中如何调用webservice的实例参考
2013/04/25 PHP
解析thinkphp中的M()与D()方法的区别
2013/06/22 PHP
php jsonp单引号转义
2014/11/23 PHP
PHP sleep()函数, usleep()函数
2016/08/25 PHP
JavaScript 学习笔记(五)
2009/12/31 Javascript
jquery中常用的SET和GET$(”#msg”).html循环介绍
2013/10/09 Javascript
js鼠标及对象坐标控制属性详细解析
2013/12/14 Javascript
jquery $(&quot;#variable&quot;) 循环改变variable的值示例
2014/02/23 Javascript
图片放大镜jquery.jqzoom.js使用实例附放大镜图标
2014/06/19 Javascript
node.js中的url.format方法使用说明
2014/12/10 Javascript
了不起的node.js读书笔记之例程分析
2014/12/22 Javascript
JavaScript实现给按钮加上双重动作的方法
2015/08/14 Javascript
通过Tabs方法基于easyUI+bootstrap制作工作站
2016/03/28 Javascript
实例讲解JavaScript中instanceof运算符的用法
2016/06/08 Javascript
JS 根据子网掩码,网关计算出所有IP地址范围示例
2020/04/23 Javascript
javascript 初学教程及五子棋小程序的简单实现
2017/07/04 Javascript
如何快速解决JS或Jquery ajax异步跨域的问题
2018/01/08 jQuery
微信小程序request请求封装,验签代码实例
2019/12/04 Javascript
Vue组件化开发之通用型弹出框的实现
2020/02/28 Javascript
Object.keys() 和 Object.getOwnPropertyNames() 的区别详解
2020/05/21 Javascript
[50:28]2018DOTA2亚洲邀请赛 3.31 小组赛 A组 Newbee vs KG
2018/04/01 DOTA
Python读取视频的两种方法(imageio和cv2)
2018/04/15 Python
Python面向对象基础入门之编码细节与注意事项
2018/12/11 Python
详解Python数据可视化编程 - 词云生成并保存(jieba+WordCloud)
2019/03/26 Python
python实现在函数图像上添加文字和标注的方法
2019/07/08 Python
python利用tkinter实现屏保
2019/07/30 Python
python实现吃苹果小游戏
2020/03/21 Python
wedgwood加拿大官网:1759年成立的英国国宝级陶瓷餐具品牌
2018/07/17 全球购物
怎么可以提高数据库查询数据的速度
2014/06/28 面试题
大学生创业计划书的用途
2014/01/08 职场文书
弘扬雷锋精神活动演讲稿
2014/03/04 职场文书
安全责任书范文
2014/03/12 职场文书
党员一句话承诺大全
2014/03/28 职场文书
万能检讨书开头与结尾怎么写
2015/02/17 职场文书
关于maven依赖 ${xxx.version}报错问题
2022/01/18 Java/Android