`
holdbelief
  • 浏览: 707539 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

闭 包

阅读更多

闭包

我没有自觉地学习过JavaScript。我必须快点了解它,因为我发现如果没有它,在实际工作中编写AJAX应用程序的准备就不会充分。开始,我感到我的编程水平好像降了几个级别。(JavaScript!我的C++朋友会怎么说?)但一旦我克服最初的障碍,我就发现JavaScript实际上是功能强大、表现力极强而且非常简练的语言。它甚至具有其它更流行的语言才刚刚开始支持的功能。

 

JavaScript的更高级功能之一是它支持闭包,这是C#2.0通过它的匿名方法支的功能。闭包是当内部函数(或C#中的内部匿名方法)绑定到它们外部函数的本地变量时所发生的运行时现象。很明显,除非此内部函数以某种方式被外部函数访问,否则它没有多少意义。实例可以更好说明这一点。

 

假设需要根据一个简单条件一个数字序列,这个条件是:只有大于100的数字才能通过筛选,并忽略其余数字。为此,可以编写类似图8中的函数。

Figure 8 根据谓词筛选元素

<script>
function filter(pred, arr) 
{
    var len = arr.length;
    var filtered = []; // shorter version of new Array();
    // iterate through every element in the array...
    for(var i = 0; i < len; i++) 
    {
        var val = arr[i];
        // if the element satisfies the predicate let it through
        if(pred(val)) 
        {
            filtered.push(val);
        }
    }
    return filtered;
}

var someRandomNumbers = [12, 32, 1, 3, 2, 2, 234, 236, 632,7, 8];
var numbersGreaterThan100 = filter
(
    function(x) 
    { 
          return (x > 100) ? true : false; 
    }, 
    someRandomNumbers
);
// displays 234, 236, 632
alert(numbersGreaterThan100);
</script>

 

但是,现在要创建不同的筛选条件,假设这次只有大于300的数字才能通过筛选,则可以编写下面这样的函数:

var greaterThan300 = filter
(
    function(x) 
    { 
         return (x > 300) ? true : false; 
    }, 
    someRandomNumbers
);

 然后,也许需要筛选大于50、25、10、600 如此等等的数字,但作为一个聪明人,您会发现它们全部都有相同的谓词"greater than",只有数字不同。因此,可以用类似下面的函数分开各个数字:

function makeGreaterThanPredicate(lowerBound) 
{
    return function(numberToCheck) 
    {
        return (numberToCheck > lowerBound) ? true : false;
    };
}

 这样,您就可以编写以下代码:

var greaterThan10 = makeGreaterThanPredicate(10);
var greaterThan100 = makeGreaterThanPredicate(100);
alert(filter(greaterThan10, someRandomNumbers));
alert(filter(greaterThan100, someRandomNumbers));

 通过观察函数makeGreaterThanPredicate返回的内部匿名函数,可以发现,该匿名内部函数使用lowerBound,后者是传递给makeGreaterThanPredicate的参数。按照作用域的一般规则,当makeGreaterThanPredicate退出时,lowerBound超出了作用域!但在这里,内部匿名函数仍然携带lowerBound,甚至在makeGreaterThanPredicate退出之后的很长时间内仍然如此。这就是我们所说的闭包;因为内部函数关闭定义它的环境(即外部函数的参数和本地变量)。

  开始可能感觉不到闭包的功能很强大。但如果应用恰当,它们就可以非常有创造性地帮您将想法转换成代码,这个过程非常有趣。在JavaScript中,闭包最有趣的用途之一是模拟类的私有变量。

下一节:模拟私有属性

分享到:
评论
Global site tag (gtag.js) - Google Analytics