本文面向初学者,大神轻喷。
好了,开始吧。
上一节 JavaScript: 零基础轻松学闭包(1)中,我们对闭包的原理进行了讲解,这一节会说很多实战性的东西了,可能会有点难度,你准备好了吗?
还记得在上一节中,有这样一个例子么?
var test = function(){
var i = 10;
}
function test2(){
alert(i);
}
test2();
函数 test 和 test2 各自形成一个闭包,两个闭包之间无法访问对方的私有数据。比如,在 test 中定义的变量,在 test2 里面是无法直接访问到的。
那么问题来了, 当然,这边和挖掘机没关系。这里的问题是,有没有什么办法让 test2 可以访问到其他闭包中的私有变量呢?
办法当然是有的,最直接的想法就是,大不了我定义一个全局变量,在 test 中将私有数据赋给全局变量,然后在 test2 里面就能访问到了。
是的,因为两个函数共同享有一个全局作用域,所以这个办法确实可行。我在很多项目里也的确看到很多人就是这么做的。
那么,有没有一种更好的方法呢?要知道,全局作用域是一个比较敏感的地方,一不小心就会出现变量名重复的问题。顺便说一句,在全局作用域中,尽量不要使用诸如 temp , a , b , c 这一类的大众化变量。
于是,这就牵扯到返回值的相关知识了,你在C语言的教材中肯定见惯了类似于这样的代码
int sum(int a,int b)
{
return a + b;
}
int all = sum(3,5);
这是一个简单的求和函数,很多人慢慢地养成了这样一个观念,就是函数的返回值就是一个字面值,要么是数字类型,要么是布尔类型,或者是字符串。
在很多强类型的语言,诸如 Java,C,C++, 确实如此。但是 return 在 JavaScript 中却大有来头。
在上一节已经说明了,js 的函数也是一种数据类型,你可以把函数看成是和int , float , double 一样的东西。
那么,既然int可以当做函数的参数或者返回值,函数当然也可以!
请看下面两句话:
在js中
这一章不讲回调函数,如果你不清楚啥叫回调函数,可以去看看这个小例子:
(浅谈js回调函数)
还是上面的那个例子,我们希望在 test2 中可以访问到 test 里面的变量,可以这样做:
var test = function(){
var i = 10;
/* 定义一个函数将变量i暴露出去 */
var get = function(){
return i ;
}
return get; //将获得i的函数暴露出去
}
function test2(){
var fn= test();//接收test暴露出来的函数
alert(fn()); //获得test中的私有数据
}
test2();
Paste_Image.png
test 函数中的 get 方法是一个内部函数,它自己也形成了一个闭包, test 是他的父级作用域,因此它可以获取i的值。
i 进入 get 方法的闭包,被包了起来,然后最终被返回了出去。
而对于 test2 来说,是可以访问到 test函数的,因此可以调用并执行 test 函数,从而获取其返回值。
你可能会说,我直接在test中把i给return出去就好了嘛,干嘛这么麻烦。
是的,言之有道理。
可是,如果我要访问 test 中多个私有数据咋办捏?
这下你可明白了吧!
现在,我们给出关于闭包的第二个注解:
(第一个注解在上一节)
Paste_Image.png
通过这张图,是不是好理解一些了呢?我们这一节单说函数里的私有数据。
刚才的例子说明,在js中,return出去的可以是基本数据类型,也可以是函数类型。
其实,JavaScript是一种基于对象的语言,也有对象的概念,所以,我们可以把你需要的东西包裹成一个对象返回出去!
上代码:
var test = function(){
var apple = '苹果';
var pear = '梨子';
/* 定义一个函数将水果暴露出去 */
var getFruit = {
apple : apple ,
pear : pear
}
return getFruit; //将获得i的函数暴露出去
}
function test2(){
var getFruit = test();//接收test暴露出来的函数
console.log(getFruit);
}
test2();
像这样用 { } 括起来的东西就是一个js对象,也就是所谓json。你可能经常会听到json这个词,觉得还挺高大上的。其实它就是一个用 { } 包起来的数据而已。
里面是键值对的形式,非常类似于Java里面的HashMap。
在这个例子中,我们可以直接把需要暴露的私有数据用一个 { } 包起来,构成一个json对象return出去就可以啦。
因为是 js 对象,alert 不能看到里面的具体内容,所以我们使用 console.log() ,结果如下:
Paste_Image.png
展开后:
Paste_Image.png
这样是不是也可以了?多出来的 proto 是原型链,以后会讲到。
大家都还记得西游记里孙悟空用遮天的把戏骗来的紫金葫芦吗,只要你拿着这个葫芦,叫一声别人的名字,如果答应了,别人就会被吸进去。
OK,这个紫金葫芦里面不正如一个闭包吗?
对不对嘛,所以,我们用闭包的知识来做一个好玩的东西吧。
<body>
<div id='box' style='width:50px;height:50px;background:#333;color:#fff;text-align:center;line-height:50px'>小妖</div>
</body>
Paste_Image.png
紫金葫芦里面的源码大概是这样的:
var 紫金葫芦 = function(id){
var domElement = document.getElementById(id);
var returnObject = {
domElement : domElement ,
backgroundColor : function(color){
domElement.style.backgroundColor = color;
},
click : function(fn){
domElement.onclick = fn;
}
};
return returnObject;
}
注:我纯粹是为了看起来方便而采用中文定义变量,在实际开发中,千万不要使用中文变量。
我们在返回出去的对象上加了三个东西:
1.domElement
你传进来一个id,我就用 document.getElementById 来包一下,得到一个dom元素,最终要操作的也就是这个dom元素。也就是说:
var box1 = 紫金葫芦('box').domElement;
var box2 = document.getElementById('box');
alert(box1 === box2);
Paste_Image.png
他们是一个东西,一样的。
紫金葫芦('box');
这行代码一旦执行,紫金葫芦就会返回 returnObject 对象,也就是说。我们喊一声 “box”,那个id为box的小妖一答应,就被装进来了,然后我们可以对它为所欲为!
比如,给它换一个背景色:
2.backgroundColor 给元素添加背景色的方法
var box = 紫金葫芦('box');
box.backgroundColor('red');
Paste_Image.png
3.click 给元素添加点击事件,需要传入一个回调函数
var box = 紫金葫芦('box');
box.click(function(){
alert('就没人吐槽这个无聊的作者么,小妖也有尊严的好么,啊喂!!');
});
结果:
click.gif
也许你已经发现了,这些方法是不是和jQuery有点类似呢?
本章结束 ...
剽悍一小兔,电气自动化毕业。
参加工作后对计算机感兴趣,深知初学编程之艰辛。
希望将自己所学记录下来,给初学者一点帮助。
免责声明: 博客中所有的图片素材均来自百度搜索,仅供学习交流,如有问题请联系我,侵立删,谢谢。