ecmascript 的函数实际上是功能完整的对象。
ecmascript 最令人感兴趣的可能莫过于函数实际上是功能完整的对象。
function 类可以表示开发者定义的任何函数。
用 function 类直接创建函数的语法如下:
var function_name = new function(arg1, arg2, ..., argn, function_body)
在上面的形式中,每个 arg 都是一个参数,最后一个参数是函数主体(要执行的代码)。这些参数必须是字符串。
记得下面这个函数吗?
function sayhi(sname, smessage) { alert("hello " + sname + smessage); }
还可以这样定义它:
var sayhi = new function("sname", "smessage", "alert(\"hello \" + sname + smessage);");
虽然由于字符串的关系,这种形式写起来有些困难,但有助于理解函数只不过是一种引用类型,它们的行为与用 function 类明确创建的函数行为是相同的。
请看下面这个例子:
function doadd(inum) {
alert(inum + 20);
}
function doadd(inum) {
alert(inum + 10);
}
doadd(10); //输出 "20"
如你所知,第二个函数重载了第一个函数,使 doadd(10) 输出了 "20",而不是 "30"。
如果以下面的形式重写该代码块,这个概念就清楚了:
var doadd = new function("inum", "alert(inum + 20)"); var doadd = new function("inum", "alert(inum + 10)"); doadd(10);
请观察这段代码,很显然,doadd 的值被改成了指向不同对象的指针。函数名只是指向函数对象的引用值,行为就像其他对象一样。甚至可以使两个变量指向同一个函数:
var doadd = new function("inum", "alert(inum + 10)"); var alsodoadd = doadd; doadd(10); //输出 "20" alsodoadd(10); //输出 "20"
在这里,变量 doadd 被定义为函数,然后 alsodoadd 被声明为指向同一个函数的指针。用这两个变量都可以执行该函数的代码,并输出相同的结果 - "20"。因此,如果函数名只是指向函数的变量,那么可以把函数作为参数传递给另一个函数吗?回答是肯定的!
function callanotherfunc(fnfunction, vargument) {
fnfunction(vargument);
}
var doadd = new function("inum", "alert(inum + 10)");
callanotherfunc(doadd, 10); //输出 "20"
在上面的例子中,callanotherfunc() 有两个参数 - 要调用的函数和传递给该函数的参数。这段代码把 doadd() 传递给 callanotherfunc() 函数,参数是 10,输出 "20"。
注意:尽管可以使用 function 构造函数创建函数,但最好不要使用它,因为用它定义函数比用传统方式要慢得多。不过,所有函数都应看作 function 类的实例。
如前所述,函数属于引用类型,所以它们也有属性和方法。
ecmascript 定义的属性 length 声明了函数期望的参数个数。例如:
function doadd(inum) { alert(inum + 10); } function sayhi() { alert("hi"); } alert(doadd.length); //输出 "1" alert(sayhi.length); //输出 "0"
函数 doadd() 定义了一个参数,因此它的 length 是 1;sayhi() 没有定义参数,所以 length 是 0。
记住,无论定义了几个参数,ecmascript 可以接受任意多个参数(最多 25 个),这一点在《函数概述》这一章中讲解过。属性 length 只是为查看默认情况下预期的参数个数提供了一种简便方式。
function 对象也有与所有对象共享的 valueof() 方法和 tostring() 方法。这两个方法返回的都是函数的源代码,在调试时尤其有用。例如:
function doadd(inum) { alert(inum + 10); } document.write(doadd.tostring());
上面这段代码输出了 doadd() 函数的文本。亲自试一试!