在JavaScript中,函數(shù)聲明和函數(shù)表達式都是使用function關鍵字來創(chuàng)建函數(shù)的,是不是覺得它們很像,很容易混淆。下面本篇文章就來帶大家分析一下函數(shù)表達式和函數(shù)聲明,介紹一下函數(shù)表達式和函數(shù)聲明的區(qū)別。
在 JavaScript 中,function
關鍵字可以完成一個簡單的工作:創(chuàng)建一個函數(shù)。 但是,使用關鍵字定義函數(shù)的方式可以創(chuàng)建具有不同屬性的函數(shù)。
在本文中,我們來看一下,如何使用function
關鍵字來定義函數(shù)聲明和函數(shù)表達式,以及這兩種函數(shù)之間的區(qū)別又是什么。
1.函數(shù)表達式vs函數(shù)聲明
函數(shù)聲明和函數(shù)表達式是使用 function
關鍵字創(chuàng)建函數(shù)的2種方法。
舉個例子來說明差異,我們創(chuàng)建兩個版本的 sums 函數(shù):
function sumA(a, b) { return a + b; } (function sumB(a, b) { return a + b; }); sumA(1, 2); // ??? sumB(1, 2); // ???
動手試試:https://jsfiddle.net/dmitri_pavlutin/8b46yokr/2/
一般情況,像往常一樣定義函數(shù)(sumA函數(shù)
)。在另一種情況下,函數(shù)被放置在一對括號中(sumB函數(shù)
)。
如果調用 sumA(1,2)
和 sumB(1,2)
會發(fā)生什么?
如預期的那樣,sumA(1, 2)
返回 3
。但是,調用sumB(1, 2)
會引發(fā)異常:Uncaught ReferenceError: sumB is not defined
。
其原因是sumA
是使用函數(shù)聲明創(chuàng)建的,該函數(shù)聲明在當前作用域中創(chuàng)建一個函數(shù)變量(具有與函數(shù)名稱相同的名稱)。 但是sumB
是使用函數(shù)表達式創(chuàng)建的(將其包裝在括號中),該函數(shù)表達式不會在當前作用域內創(chuàng)建函數(shù)變量。
如果你想訪問使用函數(shù)表達式創(chuàng)建的函數(shù),那么將函數(shù)對象保存到一個變量中:
// Works! const sum = (function sumB(a, b) { return a + b; }); sum(1, 2); // => 3
如果語句以`function
`關鍵字開頭,則為函數(shù)聲明,否則為函數(shù)表達式。
// 函數(shù)聲明:以`function`關鍵字開頭 function sumA(a, b) { return a + b; } // 函數(shù)表達式:不以`function`關鍵字開頭 const mySum = (function sumB(a, b) { return a + b; }); // 函數(shù)表達式:不以`function`關鍵字開頭 [1, 2, 3].reduce(function sum3(acc, number) { return acc + number });
從更高的角度來看,函數(shù)聲明對于創(chuàng)建獨立函數(shù)很有用,但是函數(shù)表達式可以用作回調。
現(xiàn)在,我們更深入地研究函數(shù)聲明和函數(shù)表達式的行為。
2.函數(shù)聲明
在前面的示例中已經看到的,sumA
是一個函數(shù)聲明:
// Function declaration function sumA(a, b) { return a + b; } sumA(4, 5); // => 9
當一個語句包含function
關鍵字,后跟函數(shù)名稱,一對帶參數(shù)的括號(param1, param2, paramN)
以及包圍在一對花括號{}
中的函數(shù)主體時,就會發(fā)生函數(shù)聲明。
函數(shù)聲明會創(chuàng)建一個函數(shù)變量:一個與函數(shù)名稱同名的變量(例如,上一個示例中的sumA
)。 在當前作用域中(在函數(shù)聲明之前和之后),甚至在函數(shù)作用域本身內,都可以訪問該函數(shù)變量。
函數(shù)變量通常用于調用函數(shù)或將函數(shù)對象傳遞給其他函數(shù)(傳遞給高階函數(shù))。
例如,編寫一個函數(shù) sumArray(array)
,以遞歸方式累加一個數(shù)組的項(該數(shù)組可以包含數(shù)字或其他數(shù)組):
sumArray([10, [1, [5]]]); // => 16 function sumArray(array) { let sum = 0; for (const item of array) { sum += Array.isArray(item) ? sumArray(item) : item; } return sum; } sumArray([1, [4, 6]]); // => 11
動手試試:https://jsfiddle.net/dmitri_pavlutin/n7wcryuo/
function sumArray(array) { ... }
是函數(shù)聲明。
包含函數(shù)對象的函數(shù)變量sumArray
在當前作用域中可用:sumArray([10, [1, [5]]])
之前和sumArray([1, [4, 6]])
之后,函數(shù)聲明, 以及函數(shù)本身的作用域sumArray([1, [4, 6]])
(允許遞歸調用)。
由于提升,函數(shù)變量在函數(shù)聲明之前可用。
2.1 函數(shù)聲明的注意事項
函數(shù)聲明語法的作用是創(chuàng)建獨立函數(shù)。 函數(shù)聲明應在全局作用域內,或直接在其他函數(shù)的作用域內:
// Good! function myFunc1(param1, param2) { return param1 + param2; } function bigFunction(param) { // Good! function myFunc2(param1, param2) { return param1 + param2; } const result = myFunc2(1, 3); return result + param; }
基于相同的原因,不建議在條件(if
)和循環(huán)(while
,for
)中使用函數(shù)聲明:
// Bad! if (myCondition) { function myFunction(a, b) { return a * b; } } else { function myFunction(a, b) { return a + b; } } myFunction(2, 3);
使用函數(shù)表達式更好地執(zhí)行有條件地創(chuàng)建函數(shù)。
3.函數(shù)表達式
當function
關鍵字在表達式內部創(chuàng)建函數(shù)(帶有或不帶有名稱)時,將出現(xiàn)函數(shù)表達式。
以下是使用表達式創(chuàng)建的函數(shù)的示例:
// Function expressions const sum = (function sumB(a, b) { return a + b; }); const myObject = { myMethod: function() { return 42; } }; const numbers = [4, 1, 6]; numbers.forEach(function callback(number) { console.log(number); // logs 4 // logs 1 // logs 1 });
在函數(shù)表達式中創(chuàng)建了兩種函數(shù):
- 如果表達式中的函數(shù)沒有名稱,例如
function(){return 42}
,那是一個匿名函數(shù)表達式 - 如果函數(shù)具有名稱,例如 上一個示例中的
sumB
和回調,那么這是一個命名函數(shù)表達式
3.1 函數(shù)表達式的注意事項
函數(shù)表達式適合作為條件創(chuàng)建的回調或函數(shù):
// Functions created conditionally let callback; if (true) { callback = function() { return 42 }; } else { callback = function() { return 3.14 }; } // Functions used as callbacks [1, 2, 3].map(function increment(number) { return number + 1; }); // => [2, 3, 4]
如果已創(chuàng)建命名函數(shù)表達式,請注意,該函數(shù)變量僅在創(chuàng)建的函數(shù)作用域內可用:
const numbers = [4]; numbers.forEach(function callback(number) { console.log(callback); // logs function() { ... } }); console.log(callback); // ReferenceError: callback is not defined
試一試:https://jsfiddle.net/dmitri_pavlutin/sujwmp10/2/
callback
是一個命名的函數(shù)表達式,因此callback函數(shù)變量僅在callback()
函數(shù)使用域可用,而在外部則不可用。
但是,如果將函數(shù)對象存儲到常規(guī)變量中,則可以在函數(shù)作用域內外從該變量訪問函數(shù)對象:
const callback = function(number) { console.log(callback); // logs function() { ... } }; const numbers = [4]; numbers.forEach(callback); console.log(callback); // logs function() { ... }
試一試:https://jsfiddle.net/dmitri_pavlutin/1btmrcu2/1/
4. 總結
根據(jù)使用function
關鍵字創(chuàng)建函數(shù)的方式,可以通過兩種方法來創(chuàng)建函數(shù):函數(shù)聲明和函數(shù)表達式。
留個問題: function sum(a, b) { return a + b } + 1
是函數(shù)聲明還是函數(shù)表達式,可以在留言中說出你的答案。