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 相关文章推荐
JavaScript Distilled 基础知识与函数
Apr 07 Javascript
多种方法实现JS动态添加事件
Nov 01 Javascript
ie 7/8不支持trim的属性的解决方案
May 23 Javascript
JQuery判断radio是否选中并获取选中值的示例代码
Oct 17 Javascript
每天一篇javascript学习小结(基础知识)
Nov 10 Javascript
js实现对ajax请求面向对象的封装
Jan 08 Javascript
jQuery实现的左右移动焦点图效果
Jan 14 Javascript
基于RequireJS和JQuery的模块化编程——常见问题全面解析
Apr 14 Javascript
Jquery和BigFileUpload实现大文件上传及进度条显示
Jun 27 Javascript
javascript checkbox/radio onchange不能兼容ie8处理办法
Jun 13 Javascript
激动人心的 Angular HttpClient的源码解析
Jul 10 Javascript
Vue-Element-Admin集成自己的接口实现登录跳转
Jun 23 Vue.js
浅谈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
安健A254立体声随身听的分析与打磨
2021/03/02 无线电
让Nginx支持ThinkPHP的URL重写和PATHINFO的方法分享
2011/08/08 PHP
php中 ob_start等函数截取标准输出的方法
2015/06/22 PHP
php递归函数三种实现方法及如何实现数字累加
2015/08/07 PHP
Yii2实现UploadedFile上传文件示例
2017/02/15 PHP
PHP面向对象程序设计之对象克隆clone和魔术方法__clone()用法分析
2019/06/12 PHP
2007/12/23更新创意无限,简单实用(javascript log)
2007/12/24 Javascript
获取客户端电脑日期时间js代码(jquery)
2012/09/12 Javascript
jQuery中change事件用法实例
2014/12/26 Javascript
JavaScript插件化开发教程 (三)
2015/01/27 Javascript
javascript清空table表格的方法
2015/05/14 Javascript
解决jquery实现的radio重新选中的问题
2015/07/03 Javascript
JavaScript+CSS实现仿Mootools竖排弹性动画菜单效果
2015/10/14 Javascript
JS实现星星评分功能实例代码(两种方法)
2016/06/09 Javascript
AngularJS控制器之间的通信方式详解
2016/11/03 Javascript
@ResponseBody 和 @RequestBody 注解的区别
2017/03/08 Javascript
vue打包的时候自动将px转成rem的操作方法
2018/06/20 Javascript
详解element-ui设置下拉选择切换必填和非必填
2019/06/17 Javascript
操作Windows注册表的简单的Python程序制作教程
2015/04/07 Python
Python实现网站注册验证码生成类
2017/06/08 Python
利用python模拟实现POST请求提交图片的方法
2017/07/25 Python
Python 查看list中是否含有某元素的方法
2018/06/27 Python
教你利用Python玩转histogram直方图的五种方法
2018/07/30 Python
基于python实现复制文件并重命名
2020/09/16 Python
HTML5新特性之用SVG绘制微信logo
2016/02/03 HTML / CSS
linux面试题参考答案(1)
2016/01/22 面试题
电气专业应届生求职信
2013/11/01 职场文书
环保建议书600字
2014/05/14 职场文书
预防煤气中毒方案
2014/06/16 职场文书
离婚起诉书范本
2015/05/18 职场文书
创业计划书之家政服务
2019/09/18 职场文书
「我的青春恋爱物语果然有问题。-妄言录-」第20卷封面公开
2022/03/21 日漫
Python学习之时间包使用教程详解
2022/03/21 Python
sqlserver连接错误之SQL评估期已过的问题解决
2022/03/23 SQL Server
【js设计模式】SOLID五大设计原则
2022/03/24 Javascript
使用 MybatisPlus 连接 SqlServer 数据库解决 OFFSET 分页问题
2022/04/22 SQL Server