ECMAScript 最使人感興趣的可能莫過於函數其實是功能完整的對象。 函數
Function 類能夠表示開發者定義的任何函數。 spa
用 Function 類直接建立函數的語法以下: 指針
var function_name = new function(arg1, arg2, ..., argN, function_body)
在上面的形式中,每一個 arg 都是一個參數,最後一個參數是函數主體(要執行的代碼)。這些參數必須是字符串。 調試
記得下面這個函數嗎? code
function sayHi(sName, sMessage) { alert("Hello " + sName + sMessage); }
還能夠這樣定義它: 對象
var sayHi = new Function("sName", "sMessage", "alert(\"Hello \" + sName + sMessage);");
雖然因爲字符串的關係,這種形式寫起來有些困難,但有助於理解函數只不過是一種引用類型,它們的行爲與用 Function 類明確建立的函數行爲是相同的。 ip
請看下面這個例子: 開發
function doAdd(iNum) {
alert(iNum + 20);
}
function doAdd(iNum) {
alert(iNum + 10);
}
doAdd(10); //輸出 "20"
如你所知,第二個函數重載了第一個函數,使 doAdd(10) 輸出了 "20",而不是 "30"。 字符串
若是如下面的形式重寫該代碼塊,這個概念就清楚了: it
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());