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
)中使用函数声明:
小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Web前端开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频
如果你觉得这些内容对你有帮助,可以添加下面V无偿领取!(备注:前端)
React
-
介绍一下react
-
React单项数据流
-
react生命周期函数和react组件的生命周期
-
react和Vue的原理,区别,亮点,作用
-
reactJs的组件交流
-
有了解过react的虚拟DOM吗,虚拟DOM是怎么对比的呢
-
项目里用到了react,为什么要选择react,react有哪些好处
-
怎么获取真正的dom
-
选择react的原因
-
react的生命周期函数
-
setState之后的流程
-
react高阶组件知道吗?
-
React的jsx,函数式编程
-
react的组件是通过什么去判断是否刷新的
-
如何配置React-Router
-
路由的动态加载模块
-
Redux中间件是什么东西,接受几个参数
-
redux请求中间件如何处理并发
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
真正的dom
-
选择react的原因
-
react的生命周期函数
-
setState之后的流程
-
react高阶组件知道吗?
-
React的jsx,函数式编程
-
react的组件是通过什么去判断是否刷新的
-
如何配置React-Router
-
路由的动态加载模块
-
Redux中间件是什么东西,接受几个参数
-
redux请求中间件如何处理并发
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】