JavaScript中函数表达式和函数声明及函数声明与函数表达式的不同


Posted in Javascript onNovember 15, 2015

函数表达式和函数声明

在ECMAScript中,创建函数的最常用的两个方法是函数表达式和函数声明,两者期间的区别是有点晕,因为ECMA规范只明确了一点:函数声明必须带有标示符(Identifier)(就是大家常说的函数名称),而函数表达式则可以省略这个标示符:

函数声明:

function 函数名称 (参数:可选){ 函数体 }

函数表达式:

function 函数名称(可选)(参数:可选){ 函数体 }

所以,可以看出,如果不声明函数名称,它肯定是表达式,可如果声明了函数名称的话,如何判断是函数声明还是函数表达式呢?ECMAScript是通过上下文来区分的,如果function foo(){}是作为赋值表达式的一部分的话,那它就是一个函数表达式,如果function foo(){}被包含在一个函数体内,或者位于程序的最顶部的话,那它就是一个函数声明。

function foo(){} // 声明,因为它是程序的一部分
 var bar = function foo(){}; // 表达式,因为它是赋值表达式的一部分
 new function bar(){}; // 表达式,因为它是new表达式
 (function(){
  function bar(){} // 声明,因为它是函数体的一部分
 })();

还有一种函数表达式不太常见,就是被括号括住的(function foo(){}),他是表达式的原因是因为括号 ()是一个分组操作符,它的内部只能包含表达式,我们来看几个例子:

  function foo(){} // 函数声明
  (function foo(){}); // 函数表达式:包含在分组操作符内

命名函数表达式

提到命名函数表达式,理所当然,就是它得有名字,前面的例子var bar = function foo(){};就是一个有效的命名函数表达式,但有一点需要记住:这个名字只在新定义的函数作用域内有效,因为规范规定了标示符不能在外围的作用域内有效:

var f = function foo(){
  return typeof foo; // foo是在内部作用域内有效
 };
 // foo在外部用于是不可见的
 console.log(typeof foo); // "undefined"
 console.log(f()); // "function"
var f = function foo(){
return foo; // foo是在内部作用域内有效
};
// foo在外部用于是不可见的
console.log(typeof foo); // "undefined"
console.log( f()==f); // "function"
console.log(f.name);//foo

既然,这么要求,那命名函数表达式到底有啥用啊?为啥要取名?

正如我们开头所说:给它一个名字就是可以让调试过程更方便,因为在调试的时候,如果在调用栈中的每个项都有自己的名字来描述,那么调试过程就太爽了,感受不一样嘛。

ps:JS中函数声明与函数表达式的不同

Js中的函数声明是指下面的形式:

function functionName(){ 
}

这样的方式来声明一个函数,而函数表达式则是类似表达式那样来声明一个函数,如:

var functionName = function(){ 
}

可能很多朋友在看到这两一种写法时会产生疑惑,这两种写法差不多,在应用中貌似也都是可行的,那他们有什么差别呢?

 事实上,js的解析器对函数声明与函数表达式并不是一视同仁地对待的。对于函数声明,js解析器会优先读取,确保在所有代码执行之前声明已经被解析,而函数表达式,如同定义其它基本类型的变量一样,只在执行到某一句时也会对其进行解析,所以在实际中,它们还是会有差异的,具体表现在,当使用函数声明的形式来定义函数时,可将调用语句写在函数声明之前,而后者,这样做的话会报错。

Javascript 相关文章推荐
JS基于Ajax实现的网页Loading效果代码
Oct 27 Javascript
微信小程序 欢迎页面的制作(源码下载)
Jan 09 Javascript
如何选择jQuery版本 1.x? 2.x? 3.x?
Apr 01 jQuery
Angular中实现树形结构视图实例代码
May 05 Javascript
实现div内部滚动条滚动到底部和顶部的代码
Nov 15 Javascript
通过vue-cli3构建一个SSR应用程序的方法
Sep 13 Javascript
javascript的delete运算符知识点总结
Nov 19 Javascript
使用JS来动态操作css的几种方法
Dec 18 Javascript
Js视频播放器插件Video.js使用方法详解
Feb 04 Javascript
vue+ESLint 配置保存 自动格式化代码
Mar 17 Javascript
微信小程序实现弹幕墙(祝福墙)
Nov 18 Javascript
如何使用 JavaScript 操作浏览器历史记录 API
Nov 24 Javascript
浅谈JavaScript中的对象及Promise对象的实现
Nov 15 #Javascript
javascript 中的 delete及delete运算符
Nov 15 #Javascript
详解JavaScript函数对象
Nov 15 #Javascript
javascript中window.open在原来的窗口中打开新的窗口(不同名)
Nov 15 #Javascript
深入浅析JavaScript中prototype和proto的关系
Nov 15 #Javascript
apply和call方法定义及apply和call方法的区别
Nov 15 #Javascript
JavaScript和HTML DOM的区别与联系及Javascript和DOM的关系
Nov 15 #Javascript
You might like
针对初学PHP者的疑难问答(1)
2006/10/09 PHP
PHP学习笔记之字符串编码的转换和判断
2014/05/22 PHP
php结合js实现点击超链接执行删除确认操作
2014/10/31 PHP
PHP基于imap获取邮件实例
2014/11/11 PHP
PHP的Yii框架中移除组件所绑定的行为的方法
2016/03/18 PHP
PHP 数组基本操作方法详解
2016/06/17 PHP
Yii实现文章列表置顶功能示例
2016/10/18 PHP
Javascript 不能释放内存.
2006/09/07 Javascript
JavaScript改变HTML元素的样式改变CSS及元素属性
2013/11/12 Javascript
jquery制作弹窗提示窗口代码分享
2014/03/02 Javascript
三种方式获取XMLHttpRequest对象
2014/04/21 Javascript
jQuery产品间断向下滚动效果核心代码
2014/05/08 Javascript
解决JS组件bootstrap table分页实现过程中遇到的问题
2016/04/21 Javascript
浅谈几种常用的JS类定义方法
2016/06/08 Javascript
js select下拉联动 更具级联性!
2020/04/17 Javascript
详解在Vue中使用TypeScript的一些思考(实践)
2018/07/06 Javascript
vue-cli2打包前和打包后的css前缀不一致的问题解决
2018/08/24 Javascript
Angularjs实现数组随机排序的方法
2018/10/02 Javascript
在Python中使用HTML模版的教程
2015/04/29 Python
Python打印“菱形”星号代码方法
2018/02/05 Python
用python打印菱形的实操方法和代码
2019/06/25 Python
python中property属性的介绍及其应用详解
2019/08/29 Python
python GUI库图形界面开发之PyQt5动态加载QSS样式文件
2020/02/25 Python
Django如何重置migration的几种情景
2021/02/24 Python
支教自我鉴定
2014/01/18 职场文书
乡镇消防工作实施方案
2014/03/27 职场文书
高等教育学专业自荐书
2014/06/17 职场文书
党委书记个人对照检查材料
2014/09/15 职场文书
2014年数学教师工作总结
2014/12/03 职场文书
2015年体育部工作总结
2015/04/02 职场文书
毕业论文答辩演讲稿
2015/06/23 职场文书
MySQL索引失效的典型案例
2021/06/05 MySQL
为什么MySQL分页用limit会越来越慢
2021/07/25 MySQL
mongodb清除连接和日志的正确方法分享
2021/09/15 MongoDB
解析python中的jsonpath 提取器
2022/01/18 Python
MYSQL事务的隔离级别与MVCC
2022/05/25 MySQL