实例讲解避免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 相关文章推荐
javascript中方便增删改cookie的一个类
Oct 11 Javascript
jquery中ajax函数执行顺序问题之如何设置同步
Feb 28 Javascript
详解JavaScript基本类型和引用类型
Dec 09 Javascript
小程序开发实战:实现九宫格界面的导航的代码实现
Jan 19 Javascript
JavaScript解析JSON格式数据的方法示例
Jan 24 Javascript
JS中解决谷歌浏览器记住密码输入框颜色改变功能
Feb 13 Javascript
前端自动化开发之Node.js的环境搭建教程
Apr 01 Javascript
echarts鼠标覆盖高亮显示节点及关系名称详解
Mar 17 Javascript
vue中axios请求的封装实例代码
Mar 23 Javascript
解决vuex数据异步造成初始化的时候没值报错问题
Nov 13 Javascript
Vue基于localStorage存储信息代码实例
Nov 16 Javascript
js实现简易计算器小功能
Nov 18 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
相对路径转化成绝对路径
2007/04/10 PHP
关于IIS php调用com组件的权限问题
2012/01/11 PHP
关于JSON以及JSON在PHP中的应用技巧
2013/11/27 PHP
php采集内容中带有图片地址的远程图片并保存的方法
2015/01/03 PHP
PHP实现的简易版图片相似度比较
2015/01/07 PHP
Laravel5.* 打印出执行的sql语句的方法
2017/07/24 PHP
PHP切割整数工具类似微信红包金额分配的思路详解
2019/09/18 PHP
基于laravel Request的所有方法详解
2019/09/29 PHP
给Javascript数组插入一条记录的代码
2007/08/30 Javascript
JQuery live函数
2010/12/24 Javascript
jquery select(列表)的操作(取值/赋值)
2011/03/16 Javascript
JS 如果改变span标签的是否隐藏属性
2011/10/06 Javascript
JS匀速运动演示示例代码
2013/11/26 Javascript
轻松创建nodejs服务器(2):nodejs服务器的构成分析
2014/12/18 NodeJs
js实现TAB切换对应不同颜色的代码
2015/08/31 Javascript
RequireJS使用注意细节
2016/05/15 Javascript
JavaScript数据类型学习笔记分享
2016/09/01 Javascript
Vue Transition实现类原生组件跳转过渡动画的示例
2017/08/19 Javascript
Vue.js实现按钮的动态绑定效果及实现代码
2017/08/21 Javascript
基于游标的分页接口实现代码示例
2018/11/12 Javascript
nuxt中使用路由守卫的方法步骤
2019/01/27 Javascript
深入理解es6块级作用域的使用
2019/03/28 Javascript
原生JavaScript实现贪吃蛇游戏
2020/11/04 Javascript
vant 解决tab切换插件标题样式自定义的问题
2020/11/13 Javascript
[00:59]DOTA2荣耀之路1:Doom is back!weapon X!
2018/05/22 DOTA
Python判断列表是否已排序的各种方法及其性能分析
2016/06/20 Python
Python实现Selenium自动化Page模式
2019/07/14 Python
python默认参数调用方法解析
2020/02/09 Python
css3实现信纸/同学录效果的示例代码
2018/12/11 HTML / CSS
家庭睡衣和家庭用品:Little Blue House
2018/03/18 全球购物
Brother加拿大官网:打印机、贴标机、缝纫机
2019/10/09 全球购物
阿尔卡特(中国)的面试题目
2014/08/20 面试题
财务会计专业求职信范文
2013/12/31 职场文书
好人好事演讲稿
2014/09/01 职场文书
教师作风整改措施思想汇报
2014/10/12 职场文书
详解Spring Security中的HttpBasic登录验证模式
2022/03/17 Java/Android