通过示例彻底搞懂js闭包


Posted in Javascript onAugust 10, 2017

例1

function sayHello(name) 
{
 var text = 'Hello ' + name;
 var sayAlert = function() { console.log(text); }
 sayAlert();
}

sayHello("Bob") // 输出"Hello Bob"

在sayHello()函数中定义并调用了sayAlert()函数;sayAlert()作为内层函数,可以访问外层函数sayHello()中的text变量。

例2

function sayHello2(name) 
{
 var text = 'Hello ' + name; // 局部变量
 var sayAlert = function() { console.log(text); }
 return sayAlert;
}

var say2 = sayHello2("Jane");
say2(); // 输出"Hello Jane"

例3

function buildList(list) {
 var result = [];
 for(var i = 0; i < list.length; i++) {
  var item = 'item' + list[i];
  result.push( 
    function() {
      console.log(item + ' ' + list[i]);
    } 
   );
 }
 return result;
}

var fnlist = buildList([1,2,3]);
for (var j = 0; j < fnlist.length; j++) {
  fnlist[j](); 
}

得到的结果:连续输出3个"item3 undefined"

解析:通过执行buildList函数,返回了一个result,那么这个result存放的是3个匿名函数。然而这三个匿名函数其实就是三个闭包,因为它可以访问到父函数的局部变量。所以闭包内的保留的i是最终的值为3.所以list[3]肯定是undefined. item变量值为item3.

改成如下代码:

function buildList(list) {
 var result = [];
 for(var i = 0; i < list.length; i++) {
  var item = 'item' + list[i];
  result.push( 
    (function(i) {
      console.log(item + ' ' + list[i]);
    })(i)
   );
 }
 return result;
}

var fnlist = buildList([1,2,3]);

得到的结果:

item1 1
item2 2
item3 3

解释:这儿虽然传递了一个数组进去,但是返回的是三个自执行的函数。

例4

function newClosure(someNum, someRef) 
{
 var anArray = [1,2,3];
 var num = someNum;
 var ref = someRef;
 return function(x) 
 {
 num += x;
 anArray.push(num);
 console.log('num: ' + num + "; " + 'anArray ' + anArray.toString() + "; " + 'ref.someVar ' + ref.someVar);
 }
}
closure1 = newClosure(40, {someVar: "closure 1"}); 
closure2 = newClosure(1000, {someVar: "closure 2"}); 
closure1(5); // 打印"num: 45; anArray 1,2,3,45; ref.someVar closure 1"
closure2(-10); // 打印"num: 990; anArray 1,2,3,990; ref.someVar closure 2"

每次调用newClosure()都会创建独立的闭包,它们的局部变量num与ref的值并不相同。

例5

function sayAlice() 
{
 var sayAlert = function() { console.log(alice); }
 var alice = 'Hello Alice';
 return sayAlert;
}

var sayAlice2 = sayAlice();
sayAlice2(); // 输出"Hello Alice"

alice变量在sayAlert函数之后定义,这并未影响代码执行。因为返回函数sayAlice2所指向的闭包会包含sayAlice()函数中的所有局部变量,这自然包括了alice变量,因此可以正常打印”Hello Alice”。

例6

function setupSomeGlobals() {
 var num = 666;
 gAlertNumber = function() { console.log(num); }
 gIncreaseNumber = function() { num++; }
 gSetNumber = function(x) { num = x; }
}
setupSomeGlobals();
gAlertNumber(); // 输出666
gIncreaseNumber();
gAlertNumber(); // 输出667
gSetNumber(5);
gAlertNumber(); // 输出5

解释:首先gAlertNumber,gIncreaseNumber,gSetNumber是三个全局变量,并且其三个值都是匿名函数,然而这三个匿名函数本身都是闭包。他们操作的num都是保存在内存中的同一个num,所有会得出上面的结果。

以上这篇通过示例彻底搞懂js闭包就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
cssQuery()的下载与使用方法
Jan 12 Javascript
js跨域问题之跨域iframe自适应大小实现代码
Jul 17 Javascript
JS 精确统计网站访问量的实例代码
Jul 05 Javascript
IE6-IE9中tbody的innerHTML不能赋值的解决方法
Sep 26 Javascript
jQuery 重复加载错误以及修复方法
Dec 16 Javascript
JavaScript的ExtJS框架中数面板TreePanel的使用实例解析
May 21 Javascript
jQuery插件扩展操作入门示例
Jan 16 Javascript
webpack+vue-cil中proxyTable处理跨域的方法
Jul 20 Javascript
Bootstrap-table使用footerFormatter做统计列功能
Sep 07 Javascript
JS中appendChild追加子节点无效的解决方法
Oct 14 Javascript
javascript实现动态时钟的启动和停止
Jul 29 Javascript
JavaScript parseInt0.0000005打印5原理解析
Jul 23 Javascript
用js屏蔽被http劫持的浮动广告实现方法
Aug 10 #Javascript
JS实现下拉菜单列表与登录注册弹窗效果
Aug 10 #Javascript
浅谈sass在vue注意的地方
Aug 10 #Javascript
详解jQuery同步Ajax带来的UI线程阻塞问题及解决办法
Aug 09 #jQuery
Angular 2.0+ 的数据绑定的实现示例
Aug 09 #Javascript
Ionic3实现图片瀑布流布局
Aug 09 #Javascript
JavaScript闭包和回调详解
Aug 09 #Javascript
You might like
php运行出现Call to undefined function curl_init()的解决方法
2010/11/02 PHP
PHP判断表单复选框选中状态完整例子
2014/06/24 PHP
php通过隐藏表单控件获取到前两个页面的url
2014/09/09 PHP
简单实用的PHP防注入类实例
2014/12/05 PHP
php读取远程gzip压缩网页的方法
2014/12/29 PHP
Yii框架的布局文件实例分析
2019/09/04 PHP
laravel框架路由分组,中间件,命名空间,子域名,路由前缀实例分析
2020/02/18 PHP
Laravel框架下的Contracts契约详解
2020/03/17 PHP
深入理解Javascript闭包 新手版
2010/12/28 Javascript
javaScript(JS)替换节点实现思路介绍
2013/04/17 Javascript
js换图片效果可进行定时操作
2014/06/09 Javascript
JavaScript获取伪元素(Pseudo-Element)属性的方法技巧
2015/03/13 Javascript
JavaScript仿静态分页实现方法
2015/08/04 Javascript
javascript设计模式Constructor(构造器)模式
2016/08/19 Javascript
微信小程序中实现一对多发消息详解及实例代码
2017/02/14 Javascript
详解webpack 多入口配置
2017/06/16 Javascript
Vue计算属性的使用
2017/08/04 Javascript
Angularjs单选框相关的示例代码
2017/08/17 Javascript
jQuery幻灯片插件owlcarousel参数说明中文文档
2018/02/27 jQuery
Vue多系统切换实现方案
2018/06/05 Javascript
Vue 事件处理操作实例详解
2019/03/05 Javascript
微信小程序自定义单项选择器样式
2019/07/25 Javascript
jquery插件实现轮播图效果
2020/10/19 jQuery
python变量不能以数字打头详解
2016/07/06 Python
Python WXPY实现微信监控报警功能的代码
2017/10/20 Python
django foreignkey(外键)的实现
2019/07/29 Python
潘多拉意大利官方网上商城:网上选购PANDORA珠宝
2018/10/07 全球购物
西班牙香水和化妆品连锁店:Druni
2019/05/05 全球购物
上海微创软件面试题
2012/06/14 面试题
C#和SQL Server的面试题
2016/08/12 面试题
关于元旦的广播稿
2014/02/16 职场文书
期终自我鉴定
2014/02/17 职场文书
班级寄语大全
2014/04/10 职场文书
《周恩来的四个昼夜》观后思想汇报范文两篇
2014/09/10 职场文书
法人授权委托书范本
2014/09/17 职场文书
合法的离婚协议书范本
2014/10/23 职场文书