函数是由这样的方式进行声明的:关键字 function、函数名、一组参数,以及置于括号中的待执行代码。
JavaScript 函数的基本语法是这样的:
function functionName(arg0, arg1, ... argN) {
statements
}
1 arguments 对象
在函数代码中,使用特殊对象 arguments,开发者无需明确指出参数名,就能访问它们。由于arguments的特性,JavaScript的函数没有重载
执行结果如下:
2 Function 对象(类)
ECMAScript 的函数实际上是功能完整的对象。每个函数实际上都是Function类型的实例。Function 类可以表示开发者定义的任何函数。与其他引用类型一样具有属性和方法。函数名实际上是一个指向内存堆中某个函数对象的指针。
2.1 定义函数的方式:
1) 函数声明
function sayHi(sName, sMessage) { alert("Hello " + sName + sMessage); }
2) 函数表达式
var sayHi = function (sName, sMessage) { alert("Hello " + sName + sMessage); }
定义了一个变量sum并将其初始化为一个函数,注意到function关键字后面并没有函数名,这是因为在使用函数表达式定义函数,没必要使用函数名,通过变量sum即可引用函数。还要注意函数末尾有个分号,就像声明其他变量一样。
3) new构造函数
虽然这种用法也是函数表达式,但该用法不推荐。因为这种语法会导致解析两次代码(第一次是解析常规的ECMAScript代码,第二次是解析传入构造函数中的字符串),影响性能。
使用Function构造函数,构造函数可以接受任意数量的参数,但最后一个参数始终都被看成是函数体,前面的参数则枚举出了新函数的参数。
var sayHi = new Function("sName", "sMessage", "alert(\"Hello \" + sName + sMessage);");
2.2 Function 对象的 length 属性
函数属于引用类型,所以它们也有属性和方法
3 闭包
3.1 概念:
官方解释:闭包,指的是词法表示包括不被计算的变量的函数,也就是说,函数可以使用函数之外定义的变量。
阮一峰:闭包就是能够读取其他函数内部变量的函数
在Javascript语言中,只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成"定义在一个函数内部的函数"。
所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。
样例一:
var iBaseNum = 10;
function addNum(iNum1, iNum2) {
function doAdd() {
return iNum1 + iNum2 + iBaseNum;
}
return doAdd();
}
这里,函数 addNum() 包括函数 doAdd() (闭包)。内部函数是一个闭包,因为它将获取外部函数的参数 iNum1 和 iNum2 以及全局变量 iBaseNum 的值。 addNum() 的最后一步调用了 doAdd(),把两个参数和全局变量相加,并返回它们的和。这里要掌握的重要概念是,doAdd() 函数根本不接受参数,它使用的值是从执行环境中获取的。
注:理解闭包先看 四、JavaScript 作用域与作用域链
3.2 闭包的用途
1) 读取函数内部的变量
2)让这些变量的值始终保持在内存中。
在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这证明了,函数f1中的局部变量n一直保存在内存中,并没有在f1调用后被自动清除。
为什么会这样呢?原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制(garbage collection)回收
3.2 使用闭包的注意点
1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值
3.3 思考题
题一:
题二:
参考:
1 阮一峰:学习Javascript闭包(Closure)