JS中函数表达式 VS 函数声明,聊聊它们的不同点
在JavaScript中,函数声明和函数表达式都是使用function关键字来创建函数的,是不是觉得它们很像,很容易混淆。下面本篇文章就来带大家分析一下函数表达式和函数声明,介绍一下函数表达式和函数声明的区别。
在 JavaScript 中,function
关键字可以完成一个简单的工作:创建一个函数。 但是,使用关键字定义函数的方式可以创建具有不同属性的函数。
在本文中,我们来看一下,如何使用function
关键字来定义函数声明和函数表达式,以及这两种函数之间的区别又是什么。
1.函数表达式vs函数声明
函数声明和函数表达式是使用 function
关键字创建函数的2种方法。
举个例子来说明差异,我们创建两个版本的 sums 函数:
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/
一般情况,像往常一样定义函数(sumA函数
)。在另一种情况下,函数被放置在一对括号中(sumB函数
)。
如果调用 sumA(1,2)
和 sumB(1,2)
会发生什么?
如预期的那样,sumA(1, 2)
返回 3
。但是,调用sumB(1, 2)
会引发异常:Uncaught ReferenceError: sumB is not defined
。
其原因是sumA
是使用函数声明创建的,该函数声明在当前作用域中创建一个函数变量(具有与函数名称相同的名称)。 但是sumB
是使用函数表达式创建的(将其包装在括号中),该函数表达式不会在当前作用域内创建函数变量。
如果你想访问使用函数表达式创建的函数,那么将函数对象保存到一个变量中:
// Works! const sum = (function sumB(a, b) { return a + b; }); sum(1, 2); // => 3
如果语句以`function
`关键字开头,则为函数声明,否则为函数表达式。
// 函数声明:以`function`关键字开头 function sumA(a, b) { return a + b; } // 函数表达式:不以`function`关键字开头 const mySum = (function sumB(a, b) { return a + b; }); // 函数表达式:不以`function`关键字开头 [1, 2, 3].reduce(function sum3(acc, number) { return acc + number });
从更高的角度来看,函数声明对于创建独立函数很有用,但是函数表达式可以用作回调。
现在,我们更深入地研究函数声明和函数表达式的行为。
2.函数声明
在前面的示例中已经看到的,sumA
是一个函数声明:
// Function declaration function sumA(a, b) { return a + b; } sumA(4, 5); // => 9
当一个语句包含function
关键字,后跟函数名称,一对带参数的括号(param1, param2, paramN)
以及包围在一对花括号{}
中的函数主体时,就会发生函数声明。
函数声明会创建一个函数变量:一个与函数名称同名的变量(例如,上一个示例中的sumA
)。 在当前作用域中(在函数声明之前和之后),甚至在函数作用域本身内,都可以访问该函数变量。
函数变量通常用于调用函数或将函数对象传递给其他函数(传递给高阶函数)。
例如,编写一个函数 sumArray(array)
,以递归方式累加一个数组的项(该数组可以包含数字或其他数组):
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) { ... }
是函数声明。
包含函数对象的函数变量sumArray
在当前作用域中可用:sumArray([10, [1, [5]]])
之前和sumArray([1, [4, 6]])
之后,函数声明, 以及函数本身的作用域sumArray([1, [4, 6]])
(允许递归调用)。
由于提升,函数变量在函数声明之前可用。
2.1 函数声明的注意事项
函数声明语法的作用是创建独立函数。 函数声明应在全局作用域内,或直接在其他函数的作用域内:
// 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
)和循环(while
,for
)中使用函数声明:
// Bad! if (myCondition) { function myFunction(a, b) { return a * b; } } else { function myFunction(a, b) { return a + b; } } myFunction(2, 3);
使用函数表达式更好地执行有条件地创建函数。
3.函数表达式
当function
关键字在表达式内部创建函数(带有或不带有名称)时,将出现函数表达式。
以下是使用表达式创建的函数的示例:
// 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 });
在函数表达式中创建了两种函数:
- 如果表达式中的函数没有名称,例如
function(){return 42}
,那是一个匿名函数表达式 - 如果函数具有名称,例如 上一个示例中的
sumB
和回调,那么这是一个命名函数表达式
3.1 函数表达式的注意事项
函数表达式适合作为条件创建的回调或函数:
// 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]
如果已创建命名函数表达式,请注意,该函数变量仅在创建的函数作用域内可用:
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
是一个命名的函数表达式,因此callback函数变量仅在callback()
函数使用域可用,而在外部则不可用。
但是,如果将函数对象存储到常规变量中,则可以在函数作用域内外从该变量访问函数对象:
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. 总结
根据使用function
关键字创建函数的方式,可以通过两种方法来创建函数:函数声明和函数表达式。
留个问题: function sum(a, b) { return a + b } + 1
是函数声明还是函数表达式,可以在留言中说出你的答案。
英文文章地址:https://dmitripavlutin.com/javascript-function-expressions-and-declarations/
作者:Dmitri Pavlutin
更多编程相关知识,请访问:编程视频!!
以上是JS中函数表达式 VS 函数声明,聊聊它们的不同点的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

C++中的默认参数提供对函数参数指定默认值的功能,从而增强代码可读性、简洁性和灵活性。声明默认参数:在函数声明中将参数后加上"="符号,后跟默认值。用法:函数调用时,若未提供可选参数,则会使用默认值。实战案例:计算两个数之和的函数,一个参数必填,另一个可填并有默认值0。优点:增强可读性、增加灵活性、减少样板代码。注意事项:只能在声明中指定,必须位于末尾,类型必须兼容。

如何使用WebSocket和JavaScript实现在线语音识别系统引言:随着科技的不断发展,语音识别技术已经成为了人工智能领域的重要组成部分。而基于WebSocket和JavaScript实现的在线语音识别系统,具备了低延迟、实时性和跨平台的特点,成为了一种被广泛应用的解决方案。本文将介绍如何使用WebSocket和JavaScript来实现在线语音识别系

WebSocket与JavaScript:实现实时监控系统的关键技术引言:随着互联网技术的快速发展,实时监控系统在各个领域中得到了广泛的应用。而实现实时监控的关键技术之一就是WebSocket与JavaScript的结合使用。本文将介绍WebSocket与JavaScript在实时监控系统中的应用,并给出代码示例,详细解释其实现原理。一、WebSocket技

如何利用JavaScript和WebSocket实现实时在线点餐系统介绍:随着互联网的普及和技术的进步,越来越多的餐厅开始提供在线点餐服务。为了实现实时在线点餐系统,我们可以利用JavaScript和WebSocket技术。WebSocket是一种基于TCP协议的全双工通信协议,可以实现客户端与服务器的实时双向通信。在实时在线点餐系统中,当用户选择菜品并下单

如何使用WebSocket和JavaScript实现在线预约系统在当今数字化的时代,越来越多的业务和服务都需要提供在线预约功能。而实现一个高效、实时的在线预约系统是至关重要的。本文将介绍如何使用WebSocket和JavaScript来实现一个在线预约系统,并提供具体的代码示例。一、什么是WebSocketWebSocket是一种在单个TCP连接上进行全双工

JavaScript和WebSocket:打造高效的实时天气预报系统引言:如今,天气预报的准确性对于日常生活以及决策制定具有重要意义。随着技术的发展,我们可以通过实时获取天气数据来提供更准确可靠的天气预报。在本文中,我们将学习如何使用JavaScript和WebSocket技术,来构建一个高效的实时天气预报系统。本文将通过具体的代码示例来展示实现的过程。We

在C++中,函数声明和定义的顺序影响编译和链接过程。最常见的是声明在前,定义在后;也可使用“forwarddeclaration”将定义放在声明前;如果两者同时存在,编译器将忽略声明,仅使用定义。

JavaScript教程:如何获取HTTP状态码,需要具体代码示例前言:在Web开发中,经常会涉及到与服务器进行数据交互的场景。在与服务器进行通信时,我们经常需要获取返回的HTTP状态码来判断操作是否成功,根据不同的状态码来进行相应的处理。本篇文章将教你如何使用JavaScript获取HTTP状态码,并提供一些实用的代码示例。使用XMLHttpRequest
