实例讲解避免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 相关文章推荐
在IE中调用javascript打开Excel的代码(downmoon原作)
Apr 02 Javascript
使用Grunt.js管理你项目的应用说明
Apr 24 Javascript
一个简单的JS鼠标悬停特效具体方法
Jun 17 Javascript
js清空form表单中的内容示例
May 20 Javascript
JavaScript快速切换繁体中文和简体中文的方法及网站支持简繁体切换的绝招
Mar 07 Javascript
JavaScript之Map和Set_动力节点Java学院整理
Jun 29 Javascript
基于Vuejs和Element的注册插件的编写方法
Jul 03 Javascript
页面缩放兼容性处理方法(zoom,Firefox火狐浏览器)
Aug 29 Javascript
基于casperjs和resemble.js实现一个像素对比服务详解
Jan 10 Javascript
vue 2.0 购物车小球抛物线的示例代码
Feb 01 Javascript
详解vue-cli脚手架中webpack配置方法
Aug 22 Javascript
vue+webpack 更换主题N种方案优劣分析
Oct 28 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 常见郁闷问题答解
2006/11/25 PHP
UTF8编码内的繁简转换的PHP类
2009/07/09 PHP
php中实现简单的ACL 完结篇
2011/09/07 PHP
smarty模板引擎中内建函数if、elseif和else的使用方法
2015/01/22 PHP
php生成mysql的数据字典
2016/07/07 PHP
详解php用curl调用接口方法,get和post两种方式
2017/01/13 PHP
PHP实现从PostgreSQL数据库检索数据分页显示及根据条件查找数据示例
2018/06/09 PHP
PHP中单例模式的使用场景与使用方法讲解
2019/03/18 PHP
ExtJS的拖拽效果示例
2013/12/09 Javascript
js函数定时器实现定时读取系统实时连接数
2014/04/30 Javascript
js控制href内容的连接内容的变化示例
2014/04/30 Javascript
JS+CSS实现精美的二级导航效果代码
2015/09/17 Javascript
JavaScript常用基础知识强化学习
2015/12/09 Javascript
Bootstrap每天必学之滚动监听
2016/03/16 Javascript
AngularJS中使用three.js的实例详解
2017/07/21 Javascript
Vue使用mixins实现压缩图片代码
2018/03/14 Javascript
vue升级之路之vue-router的使用教程
2018/08/14 Javascript
基于React Native 0.52实现轮播图效果
2020/08/25 Javascript
记录vue做微信自定义分享的一些问题
2019/09/12 Javascript
详解react组件通讯方式(多种)
2020/05/06 Javascript
jquery插件实现轮播图效果
2020/10/19 jQuery
解决python写的windows服务不能启动的问题
2014/04/15 Python
Python合并字符串的3种方法
2015/05/21 Python
Python探索之修改Python搜索路径
2017/10/25 Python
使用Django连接Mysql数据库步骤
2019/01/15 Python
用Python实现二叉树、二叉树非递归遍历及绘制的例子
2019/08/09 Python
Python嵌入C/C++进行开发详解
2020/06/09 Python
Keras 中Leaky ReLU等高级激活函数的用法
2020/07/05 Python
详解CSS3选择器:nth-child和:nth-of-type之间的差异
2017/09/18 HTML / CSS
外企财务年会演讲稿
2014/01/03 职场文书
领导班子四风查摆对照检查材料思想汇报
2014/10/05 职场文书
文明单位创建材料
2014/12/24 职场文书
创卫工作总结2015
2015/04/22 职场文书
解读MySQL的客户端和服务端协议
2021/05/10 MySQL
Python 批量下载阴阳师网站壁纸
2021/05/19 Python
java实现web实时消息推送的七种方案
2022/07/23 Java/Android