从一道面试题分析javascript闭包

据说是一不注意就会做错的五道javascript面试题之一,我们来看看这道题长什么样

function Container( properties ) {
    var objthis = this;
    for ( var i in properties ) {
        (function(){
                var t = properties[i];
                objthis[ "get" + i ] = function() {return t;};
                objthis[ "set" + i ] = function(val) {t = val;};
        })();
    }
}  

var prop = {Name : "Jim", Age : 13};
var con = new Container(prop);
console.log(con.getName());  

con.setName("Lucy");
console.log(con.getName());
console.log(prop.Name);  

现在请写出每次console.log的输出结果:()

A: Jim,Lucy,Lucy  B: Jim,Jim,Jim C : Lucy ,Lucy,Lucy D: Jim,Lucy,Jim

如果你选了D,排除运气的成份,你可以不用往下看了,因为可以确定你是资深老手了。

如果是选的别的选项,请继续看下面的分析。

首先大概扫瞄一下整个Container函数:函数的作用应当是想要给传进来的对象的每一个属性都添加一个get和 set 的方法。

下面开始从上至下的逐行分析:

function Container( properties ) {
    var objthis = this;  // 用objthis 保存对this的引用
    for ( var i in properties ) {  //循环properties中的内容
        (function(){
          //这里就是闭包所创建的空间
          // 闭包内声明一个局部变量保存存properties中的值
                var t = properties[i];
                //在objthis对象上添加两个方法
                objthis[ "get" + i ] = function() {return t;};
                objthis[ "set" + i ] = function(val) {t = val;};
        })();  //创建一个立即执行函数,目的是形成一个闭包,因为我们知道,闭包的作用是即使外层函数调用后,由于闭包引用的关系,外层函数不会被立即回收,因此闭包函数还可以继续使用外层函数所声明的变量和方法。
    }
}  
var prop = {Name : "Jim", Age : 13};
var con = new Container(prop);
console.log(con.getName());  // Jim

con.setName("Lucy");
console.log(con.getName());  // Lucy

初看起来,没有什么问题,那这个坑到底在哪里呢?

如果我们继续打印 console.log(prop.Name); // Jim

是不是感到很惊讶?为什么 con.getName() 得到Lucy,而prop.Name 得到Jim ?

===============================================================

我们先从这个立即执行函数看起,如果省略它,那么t始终都是循环之后的值。这个立即执行函数就是为了解决这个问题的,所以这也没有什么问题。

然后就是这个objthis,它是Container实例的一个内部指针,和 properties 没有什么关系,但是properties 和prop是有关系的。因为prop是对象

是引用类型的,所以properties 可以看成是prop引用的一个副本(关于javascript的传参,后面再细说)。所以在Container内部,能过修改properties

是可以改变prop本身的,当然也可以访问到prop的属性。

我来看看

var con = new Container(prop); 这个con到底有什么内容

  1. getAge: function () {return t;}
  2. setAge: function (val) {t = val;}
  3. getName: function () {return t;}
  4. setName: function (val) {return t;}

看来con是根据prop的属性名,生成了对应的set/get方法,但是方法体内找不到properties的痕迹。里边只有一个t,而

t=properties[i], 这不就是prop.Name或prop.Age的值吗?我第一反应就是这个set方法有问题。t = val; 根据对象属性赋值的常识

var t = prop.Name, t = ‘xxx‘, prop.Name的值是肯定不会改变的。这改变的仅仅是t自己。

con.setName("Lucy");
console.log(con.getName());  // Lucy但是为什么这里又更改生效了呢?

我在这个地方郁闷了很久,我自己没有想清楚,最后还是请教了一位资深前辈才搞清楚。

原因很简单,因为con上的set和get方法,访问的都是闭包内的t的值。与properties没有半毛钱关系了。

con.setName("Lucy"); 相当于 t = "Lucy" ,而con.getName相当于 return t; 这是再普通不过的道理了。

这样也就很好的解释了,为什么 con.getName() 的值是Lucy,而prop.Name还是jim.

为什么我会产生那样的困惑呢?原因就在于没有搞清楚这个t,没有弄明白闭包的用法。那现在要怎么修改,才能实现Container最初的意图呢?

既然明白了con.setName修改的是自身的t,con.getName访问的也是自身的t,那么只人把这个t换成prop.Name就好了。

下面是我修改后的代码:

function Container( properties ) {
    var objthis = this;
    for ( var i in properties ) {
        (function(){
                var name = i;
                objthis[ "get" + i ] = function() {return properties[name];};
                objthis[ "set" + i ] = function(val) {properties[name] = val;};
        })();
    }
}

con.setName(‘frog‘)
console.log(prop.Name,con.getName());

// frog frog

======================================================================

关于javascript中的参数的传递。

根据我们c语言的知识,参数的传递有两种方式,按引用传递和按值传递。

但是javascript中所以的参数都是按值传递的。即便参数是对像也是如此。

function fun(a){

}

var o = {name:‘frog‘}

fun(o);

fun(o/*这里的o其实是复制了o对象的一个引用*/)

这不是本篇的重点,只是提一下。不明白的私信我或查阅相关资料。

时间: 2024-11-05 11:24:36

从一道面试题分析javascript闭包的相关文章

关于阿里的一道笔试题分析

其题目如下: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 #pragma pack(2) class A { public:     int i;     union U     {         char buff[13];         int i;     }u;     void foo() {    }     typedef char* (*f)(void*);     enum{red, green, blue} color; }a; class A

你不一定能做对的JavaScript闭包面试题

由工作中演变而来的面试题 这是一个我工作当中的遇到的一个问题,似乎很有趣,就当做了一道题去面试,发现几乎没人能全部答对并说出原因,遂拿出来聊一聊吧. 先看题目代码: function fun(n,o) { console.log(o) return { fun:function(m){ return fun(m,n); } }; } var a = fun(0); a.fun(1); a.fun(2); a.fun(3);//undefined,?,?,? var b = fun(0).fun(

JavaScript 闭包原理分析

本文转载至 http://www.ruanyifeng.com/blog/2009/08/learning_javascript_closures.html 另一篇很好的资料 http://www.kryogenix.org/code/browser/secrets-of-javascript-closures/secrets_of_javascript_closures.pdf 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 下面就是我

JavaScript 闭包 面试题

<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta http-equiv="X-UA-Compatible"

JavaScript闭包(Closure)学习笔记

闭包(closure)是JavaScript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 下面就是我的学习笔记,对于JavaScript初学者应该是很有用的. 一.变量的作用域 要理解闭包,首先必须理解JavaScript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. JavaScript语言的特殊之处,就在于函数内部可以直接读取全局变量. var n=999; function f1() { alert(n); } f1(); // 999 另一方面,在函数

JavaScript闭包学习笔记

原文:JavaScript闭包学习笔记 闭包(closure)是JavaScript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 下面就是我的学习笔记,对于JavaScript初学者应该是很有用的. 一.变量的作用域 要理解闭包,首先必须理解JavaScript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. JavaScript语言的特殊之处,就在于函数内部可以直接读取全局变量. 1 var n=999; 2 3 function f1() { 4 alert

JavaScript 闭包

1.词法作用域: 简单地说子集能访问父级的变量, 说人话就是变量拿来就用不用传入 2.函数局部变量: 在函数体中以var 声明变量的为局部变量 + 函数传入的参数, 直接写变量名声明的变量是全局变量 3.局部变量生存期: 局部变量在函数函数的执行期间可用,  一旦执行过后,局部变量将不再可用 4.延长局部变量生存期: 现在问题来了,我想要延长局部变量的生存期,怎么办.(因为调用函数不仅仅是为了return, 有时候还需要保存函数中的状态, 或者实现类等等) 5.使用全局变量不好吗: 不好.有时函

全面理解Javascript闭包和闭包的几种写法及用途

一.什么是闭包和闭包的几种写法和用法                                                       1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点: 1. 作为一个函数变量的一个引用,当函数返回时,其处于激活状态. 2. 一个闭包就是当一个函数返回时,一个没有释放资源的栈区. 简单的说,Javascript允许使用内部函数---即函数定义和函数表

转 全面理解Javascript闭包和闭包的几种写法及用途

转自:http://www.cnblogs.com/yunfeifei/p/4019504.html 好久没有写博客了,过了一个十一长假都变懒了,今天总算是恢复状态了.好了,进入正题,今天来说一说javascript里面的闭包吧!本篇博客主要讲一些实用的东西,主要将闭包的写法.用法和用途. 一.什么是闭包和闭包的几种写法和用法                                                       1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑