1、注意一下几段代码:
sayHi(); function sayHi() { alert("Hi!"); } sayHi(); //错误:函数还不存在 var sayHi = function () { alert("Hi!"); }; //不要这样做! if (condition) { function sayHi() { alert("Hi!"); } } else { function sayHi() { alert("Yo!"); } } //可以这样做 var sayHi; if (condition) { sayHi = function () { alert("Hi!"); }; } else { sayHi = function () { alert("Yo!"); }; }
能够创建函数再赋值给变量,也就能够把函数作为其他函数的值返回
function createComparisonFunction(propertyName) { return function (object1, object2) { var value1 = object1[propertyName]; var value2 = object2[propertyName]; if (value1 < value2) { return -1; } else if (value1 > value2) { return 1; } else { return 0; } }; }
3、递归
//递归函数 function factorial(num) { if (num <= 1) { return 1; } else { return num * factorial(num - 1); } } //但是这样使用会发生错误 var anotherFactorial = factorial; factorial = null; alert(anotherFactorial(4)); //出错! //解决方法一,在非严格模式可以这么解决 function factorial(num) { if (num <= 1) { return 1; } else { return num * arguments.callee(num - 1); } } //解决方法二:严格模式下这么写 var factorial = (function f(num) { if (num <= 1) { return 1; } else { return num * f(num - 1); } });
7.5 小结
在JavaScript 编程中,函数表达式是一种非常有用的技术。使用函数表达式可以无须对函数命名,
从而实现动态编程。匿名函数,也称为拉姆达函数,是一种使用JavaScript 函数的强大方式。以下总结
了函数表达式的特点。
函数表达式不同于函数声明。函数声明要求有名字,但函数表达式不需要。没有名字的函数表
达式也叫做匿名函数。
在无法确定如何引用函数的情况下,递归函数就会变得比较复杂;
递归函数应该始终使用arguments.callee 来递归地调用自身,不要使用函数名——函数名可
能会发生变化。
当在函数内部定义了其他函数时,就创建了闭包。闭包有权访问包含函数内部的所有变量,原理
如下。
在后台执行环境中,闭包的作用域链包含着它自己的作用域、包含函数的作用域和全局作用域。
通常,函数的作用域及其所有变量都会在函数执行结束后被销毁。
但是,当函数返回了一个闭包时,这个函数的作用域将会一直在内存中保存到闭包不存在为止。
使用闭包可以在JavaScript 中模仿块级作用域(JavaScript 本身没有块级作用域的概念),要点如下。
创建并立即调用一个函数,这样既可以执行其中的代码,又不会在内存中留下对该函数的引用。
结果就是函数内部的所有变量都会被立即销毁——除非将某些变量赋值给了包含作用域(即外
部作用域)中的变量。
闭包还可以用于在对象中创建私有变量,相关概念和要点如下。
即使JavaScript 中没有正式的私有对象属性的概念,但可以使用闭包来实现公有方法,而通过公
有方法可以访问在包含作用域中定义的变量。
有权访问私有变量的公有方法叫做特权方法。
可以使用构造函数模式、原型模式来实现自定义类型的特权方法,也可以使用模块模式、增强
的模块模式来实现单例的特权方法。
JavaScript 中的函数表达式和闭包都是极其有用的特性,利用它们可以实现很多功能。不过,因为
创建闭包必须维护额外的作用域,所以过度使用它们可能会占用大量内存。