自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

JS 中的函數(shù)表達式和函數(shù)聲明你混淆了嗎?

開發(fā) 前端
在本文中,我們來看一下,如何使用function關(guān)鍵字來定義函數(shù)聲明和函數(shù)表達式,以及這兩種函數(shù)之間的區(qū)別又是什么。

[[407614]]

在 JavaScript 中,function關(guān)鍵字可以完成一個簡單的工作:創(chuàng)建一個函數(shù)。但是,使用關(guān)鍵字定義函數(shù)的方式可以創(chuàng)建具有不同屬性的函數(shù)。

在本文中,我們來看一下,如何使用function關(guān)鍵字來定義函數(shù)聲明和函數(shù)表達式,以及這兩種函數(shù)之間的區(qū)別又是什么。

1.函數(shù)表達式vs函數(shù)聲明

函數(shù)聲明和函數(shù)表達式是使用 function 關(guān)鍵字創(chuàng)建函數(shù)的2種方法。

舉個例子來說明差異,我們創(chuàng)建兩個版本的 sums 函數(shù):

  1. function sumA(a, b) { 
  2.   return a + b; 
  3.  
  4. (function sumB(a, b) { 
  5.   return a + b; 
  6. }); 
  7.  
  8. sumA(1, 2); // ??? 
  9. sumB(1, 2); // ??? 

動手試試:https://jsfiddle.net/dmitri_pavlutin/8b46yokr/2/

一般情況,像往常一樣定義函數(shù)(sumA函數(shù))。在另一種情況下,函數(shù)被放置在一對括號中(sumB函數(shù))。

如果調(diào)用 sumA(1,2) 和 sumB(1,2) 會發(fā)生什么?

如預(yù)期的那樣,sumA(1, 2) 返回 3。但是,調(diào)用sumB(1, 2)會引發(fā)異常:Uncaught ReferenceError: sumB is not defined。

其原因是sumA是使用函數(shù)聲明創(chuàng)建的,該函數(shù)聲明在當(dāng)前作用域中創(chuàng)建一個函數(shù)變量(具有與函數(shù)名稱相同的名稱)。但是sumB是使用函數(shù)表達式創(chuàng)建的(將其包裝在括號中),該函數(shù)表達式不會在當(dāng)前作用域內(nèi)創(chuàng)建函數(shù)變量。

如果你想訪問使用函數(shù)表達式創(chuàng)建的函數(shù),那么將函數(shù)對象保存到一個變量中:

  1. // Works! 
  2. const sum = (function sumB(a, b) { 
  3.   return a + b; 
  4. }); 
  5.  
  6. sum(1, 2); // => 3 
  1. 如果語句以`function`關(guān)鍵字開頭,則為函數(shù)聲明,否則為函數(shù)表達式。 
  1. // 函數(shù)聲明:以`function`關(guān)鍵字開頭 
  2. function sumA(a, b) { 
  3.   return a + b; 
  4.  
  5. // 函數(shù)表達式:不以`function`關(guān)鍵字開頭 
  6. const mySum = (function sumB(a, b) { 
  7.   return a + b; 
  8. }); 
  9.  
  10. // 函數(shù)表達式:不以`function`關(guān)鍵字開頭 
  11. [1, 2, 3].reduce(function sum3(acc, number) {  
  12.   return acc + number  
  13. }); 

從更高的角度來看,函數(shù)聲明對于創(chuàng)建獨立函數(shù)很有用,但是函數(shù)表達式可以用作回調(diào)。

現(xiàn)在,我們更深入地研究函數(shù)聲明和函數(shù)表達式的行為。

2.函數(shù)聲明

在前面的示例中已經(jīng)看到的,sumA是一個函數(shù)聲明:

  1. // Function declaration 
  2. function sumA(a, b) { 
  3.   return a + b; 
  4.  
  5. sumA(4, 5); // => 9 

當(dāng)一個語句包含function關(guān)鍵字,后跟函數(shù)名稱,一對帶參數(shù)的括號(param1, param2, paramN)以及包圍在一對花括號{}中的函數(shù)主體時,就會發(fā)生函數(shù)聲明。

函數(shù)聲明會創(chuàng)建一個函數(shù)變量:一個與函數(shù)名稱同名的變量(例如,上一個示例中的sumA)。在當(dāng)前作用域中(在函數(shù)聲明之前和之后),甚至在函數(shù)作用域本身內(nèi),都可以訪問該函數(shù)變量。

函數(shù)變量通常用于調(diào)用函數(shù)或?qū)⒑瘮?shù)對象傳遞給其他函數(shù)(傳遞給高階函數(shù))。

例如,編寫一個函數(shù) sumArray(array),以遞歸方式累加一個數(shù)組的項(該數(shù)組可以包含數(shù)字或其他數(shù)組):

  1. sumArray([10, [1, [5]]]); // => 16 
  2.  
  3. function sumArray(array) { 
  4.   let sum = 0; 
  5.   for (const item of array) { 
  6.     sum += Array.isArray(item) ? sumArray(item) : item; 
  7.   } 
  8.   return sum
  9.  
  10. sumArray([1, [4, 6]]); // => 11 

動手試試:https://jsfiddle.net/dmitri_pavlutin/n7wcryuo/

function sumArray(array) { ... } 是函數(shù)聲明。

包含函數(shù)對象的函數(shù)變量sumArray在當(dāng)前作用域中可用:sumArray([10, [1, [5]]])之前和sumArray([1, [4, 6]])之后,函數(shù)聲明, 以及函數(shù)本身的作用域sumArray([1, [4, 6]])(允許遞歸調(diào)用)。

由于提升,函數(shù)變量在函數(shù)聲明之前可用。

2.1 函數(shù)聲明的注意事項

函數(shù)聲明語法的作用是創(chuàng)建獨立函數(shù)。函數(shù)聲明應(yīng)在全局作用域內(nèi),或直接在其他函數(shù)的作用域內(nèi):

  1. // Good! 
  2. function myFunc1(param1, param2) { 
  3.   return param1 + param2; 
  4.  
  5. function bigFunction(param) { 
  6.   // Good! 
  7.   function myFunc2(param1, param2) { 
  8.     return param1 + param2; 
  9.   } 
  10.  
  11.   const result = myFunc2(1, 3); 
  12.   return result + param; 

基于相同的原因,不建議在條件(if)和循環(huán)(while,for)中使用函數(shù)聲明:

  1. // Bad! 
  2. if (myCondition) { 
  3.   function myFunction(a, b) { 
  4.     return a * b; 
  5.   } 
  6. else { 
  7.   function myFunction(a, b) { 
  8.     return a + b; 
  9.   } 
  10.  
  11. myFunction(2, 3); 

使用函數(shù)表達式更好地執(zhí)行有條件地創(chuàng)建函數(shù)。

3.函數(shù)表達式

當(dāng)function關(guān)鍵字在表達式內(nèi)部創(chuàng)建函數(shù)(帶有或不帶有名稱)時,將出現(xiàn)函數(shù)表達式。

以下是使用表達式創(chuàng)建的函數(shù)的示例:

  1. // Function expressions 
  2.  
  3. const sum = (function sumB(a, b) { 
  4.   return a + b; 
  5. }); 
  6.  
  7. const myObject = { 
  8.   myMethod: function() { 
  9.     return 42; 
  10.   } 
  11. }; 
  12.  
  13. const numbers = [4, 1, 6]; 
  14. numbers.forEach(function callback(number) { 
  15.   console.log(number); 
  16.   // logs 4 
  17.   // logs 1 
  18.   // logs 1 
  19. }); 

在函數(shù)表達式中創(chuàng)建了兩種函數(shù):

  • 如果表達式中的函數(shù)沒有名稱,例如 function(){return 42},那是一個匿名函數(shù)表達式
  • 如果函數(shù)具有名稱,例如 上一個示例中的sumB和回調(diào),那么這是一個命名函數(shù)表達式

3.1 函數(shù)表達式的注意事項

函數(shù)表達式適合作為條件創(chuàng)建的回調(diào)或函數(shù):

  1. // Functions created conditionally 
  2. let callback; 
  3. if (true) { 
  4.   callback = function() { return 42 }; 
  5. else { 
  6.   callback = function() { return 3.14 }; 
  7.  
  8. // Functions used as callbacks 
  9. [1, 2, 3].map(function increment(number) { 
  10.   return number + 1; 
  11. }); // => [2, 3, 4] 

如果已創(chuàng)建命名函數(shù)表達式,請注意,該函數(shù)變量僅在創(chuàng)建的函數(shù)作用域內(nèi)可用:

  1. const numbers = [4]; 
  2. numbers.forEach(function callback(number) { 
  3.   console.log(callback); // logs function() { ... } 
  4. }); 
  5.  
  6. console.log(callback); // ReferenceError: callback is not defined 

試一試:https://jsfiddle.net/dmitri_pavlutin/sujwmp10/2/

callback是一個命名的函數(shù)表達式,因此callback函數(shù)變量僅在callback()函數(shù)使用域可用,而在外部則不可用。

但是,如果將函數(shù)對象存儲到常規(guī)變量中,則可以在函數(shù)作用域內(nèi)外從該變量訪問函數(shù)對象:

  1. const callback = function(number) { 
  2.   console.log(callback); // logs function() { ... } 
  3. }; 
  4.  
  5. const numbers = [4]; 
  6. numbers.forEach(callback); 
  7. console.log(callback); // logs function() { ... } 

試一試:https://jsfiddle.net/dmitri_pavlutin/1btmrcu2/1/

4. 總結(jié)

根據(jù)使用function關(guān)鍵字創(chuàng)建函數(shù)的方式,可以通過兩種方法來創(chuàng)建函數(shù):函數(shù)聲明和函數(shù)表達式。

作者:Dmitri Pavlutin

譯者:前端小智 來源:dmitripavlutin原文:https://dmitripavlutin.com/javascript-function-expressions-and-declarations/

 

責(zé)任編輯:姜華 來源: 大遷世界
相關(guān)推薦

2011-05-30 16:11:46

Javascript

2017-08-01 00:19:15

Javascript函數(shù)函數(shù)聲明

2025-03-12 08:45:15

函數(shù)聲明函數(shù)表達式IIFE

2020-10-16 10:07:03

Lambda表達式Java8

2010-09-10 15:20:11

SQL函數(shù)計算表達式

2010-11-16 14:53:02

Oracle游標(biāo)表達式

2023-11-02 18:45:00

Rust編程表達式

2011-05-11 17:40:30

PHP正則表達式

2020-10-16 06:40:25

C++匿名函數(shù)

2021-07-28 15:28:22

JS逗號表達式面試題

2009-08-31 17:11:37

Lambda表達式

2023-12-24 22:30:05

LambdaJava函數(shù)

2012-04-28 15:22:46

PHP

2021-08-31 07:19:41

Lambda表達式C#

2024-12-02 10:56:29

2009-08-10 10:06:10

.NET Lambda

2009-07-21 14:03:00

Scalaif表達式while循環(huán)

2009-09-16 09:58:53

PHP正則表達式函數(shù)

2009-10-12 10:11:08

Lambda表達式編寫

2009-09-16 10:43:22

PHP正則表達式函數(shù)
點贊
收藏

51CTO技術(shù)棧公眾號