JavaScript的“闭包”到底是什么

在JavaScripot函数闭包的定义中,一般都有一个outer 函数,一个inner函数。那么“闭包”到底是指outer函数呢,还是指inner函数?

从官方定义来看,并不清楚:A closure is a combination of a code block and data of a context in which this code block is created.

 

那么在下列函数中:

 

function outerfunc(y) {

 

          var internal1 = 100 ;

          var internal2 = 200 ;

         

          function innerfunc1(x) {

                    internal1++;

                    internal2 --;

          }

                   

          function innerfunc2()

          {

                    internal1--;

                    internal2++;

          }

         

          function get1()

          {

                    return internal1;

          }

         

          function get2()

          {

                    return internal2;

          }

         

          y.func1 = innerfunc1;

          y.func2 = innerfunc2;

          y.get1 = get1;

          y.get2 = get2;

}

 

到底是哪个函数闭包那个呢?是outerfunc闭包innerFunc1, innerfunc2呢?还是相反?

一般的教科书认为是innerfunc1 闭包了internal1和internal2。同样innerfunc2闭包了上述两个变量。这也不违反上述官方定义,只是总觉得在“语义”层面说不过去。看过我C++博客的读者会知道我对语义的痴迷。

 

我认为,正确的理解应该是outerfunc, innerfunc1, innerfunc2,get1,get2共同组成一个闭包,包揽了outerfunc的所有自由变量,参数, 以及local变量。

 

为什么呢?因为:

 

1) innerfunc1 和innerfunc2 都“定义”在outerfunc内,共享outerfunc的自由变量internal1和internal2。单独谈哪一个函数都没有意义。

 

2) 因为有了innerfunc1和innerfunc2,outerfunc的“语义”发生了根本的变化。本来是“auto”变量的internal1和internal2变成了“heap”变量。具体的实现是JavaScript的implementation细节。但是懂得这一微妙的细节,会加深你对JavaScript的理解。也知道closure的实现代价。

 

3) 上面这条的详细解读:没有闭包的outerfunc可以优化,利用stack变量来实现internal1和internal2。有了闭包,internal1和internal2必须由Garbage collection 来实现。功能当然会大幅下降。

 

以上第二点,对理解“闭包”,以及JavaScript函数的性质,有根本意义。

 

2014-8-26 西雅图

 

posted @ 2014-08-26 15:08  Yang_Lian  阅读(901)  评论(3编辑  收藏  举报