JavaScript高级程序设计第5章读书笔记(12)

函数内部属性

在函数内部,有两个特殊的对象: argumentsthis。其中,arguments是一个类数组对象,包含着传入函数的所有参数。虽然arguments的主要用途是保存函数参数,但这个对象还有一个名叫callee的属性,该属性是一个指针,指向拥有这个arguments对象的函数。

定义阶乘函数一般要用到递归算法。在函数有名字,而且名字以后也不会改变的情况下,这样定义没有问题。但是,这样做,会使这个函数的执行与函数名factorial紧紧耦合在一起。为了消除这种紧密耦合的现象,可以像下面这样使用arguments.callee

像这样重写之后,无论引用函数时使用的是什么名字,都可以保证正常完成递归调用。比如:

此处,变量trueFactorial获得了factorial的值,实际上是在另一个位置上保存了一个函数的指针。然后,我们又将一个简单地返回0的函数赋值给factorial变量。如果像原来的factorial()那样不使用arguments.callee,调用trueFactorial()会返回0。可是,在解除了函数体内的代码与函数名的耦合状态之后,trueFactorial()仍然能够正常地计算阶乘。

函数内部的另一个特殊对象是this。this引用的是函数据以执行的环境对象,或者说是this值(当在网页的全局作用域中调用函数时,this对象引用的就是window)。

当在全局作用域中调用sayColor()的时候,this引用的是全局对象window,也就是说,对this.color求值会转换成对window.color求值。而当把这个函数赋给对象o并调用o.sayColor()时,this引用的是对象o。

一定要牢记,函数的名字仅仅是一个包含指针的变量而已。因此,即使在不同的环境中执行,全局的sayColor()函数与o.sayColor()指向的仍然是同一个函数。

ECMAScript 5也规范化了另一个函数对象的属性: caller。除了Opera的早期版本不支持,其他浏览器都支持这个ECMAScript 3并没有定义的属性。这个属性中保存着调用当前函数的函数的引用,如果是在全局作用域中调用当前函数,它的值是null。比如:

以上代码执行后,会在警告框中显示 outer()函数的源代码。因为outer()调用了inner(),所以inner.caller会指向outer()。

IE、Firefox、Chrome和Safari的所有版本以及Opera 9.6都支持caller属性。

当函数在严格模式下运行的时候,访问arguments.callee会导致错误。ECMAScript 5还定义了arguments.caller属性,但在严格模式下访问它也会导致错误,而在非严格模式下这个属性始终是undefined。定义这个属性是为了分清arguments.caller和函数的caller属性。以上变化都是为了加强这门语言的安全性,这样第三方代码就不能在相同的环境里窥视其他代码了。

严格模式还有一个限制就是,不能为函数的caller属性赋值,否则会导致错误。