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分页代码(当前页码居中)
Sep 20 Javascript
JS随机漂浮广告代码具体实例
Nov 19 Javascript
js实现点击左右按钮轮播图片效果实例
Jan 29 Javascript
JavaScript检测并限制复选框选中个数的方法
Aug 12 Javascript
原生js实现autocomplete插件
Apr 14 Javascript
AngularJS使用ng-class动态增减class样式的方法示例
May 18 Javascript
Vue使用枚举类型实现HTML下拉框步骤详解
Feb 05 Javascript
vue单页缓存存在的问题及解决方案(小结)
Sep 25 Javascript
JointJS流程图的绘制方法
Dec 03 Javascript
在 Vue 应用中使用 Netlify 表单功能的方法详解
Jun 03 Javascript
使用Vue实现简单计算器
Feb 25 Javascript
js实现表单项的全选、反选及删除操作示例
Jun 05 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上显示JFreechart画的统计图方法
2013/11/03 PHP
PHP中ob_start函数的使用说明
2013/11/11 PHP
php限制文件下载速度的代码
2015/10/20 PHP
php实现的双色球算法示例
2017/06/20 PHP
基于php编程规范(详解)
2017/08/17 PHP
phpcmsv9.0任意文件上传漏洞解析
2020/10/20 PHP
JS Excel读取和写入操作(模板操作)实现代码
2010/04/11 Javascript
JavaScript表达式:URL 协议介绍
2013/03/10 Javascript
JS检测输入字符是否包含非法字符的示例代码
2014/02/11 Javascript
JS for循环中i++ 和 ++i的区别介绍
2016/07/20 Javascript
jQuery内容筛选选择器实例代码
2017/02/06 Javascript
如何使用Bootstrap 按钮实例详解
2017/03/29 Javascript
vue-cli项目如何使用vue-resource获取本地的json数据(模拟服务端返回数据)
2017/08/04 Javascript
微信{"errcode":48001,"errmsg":"api unauthorized, hints: [ req_id: 1QoCla0699ns81 ]"}
2018/10/12 Javascript
优雅地使用loading(推荐)
2019/04/20 Javascript
JavaScript中将值转换为字符串的五种方法总结
2019/06/06 Javascript
Python 过滤字符串的技巧,map与itertools.imap
2008/09/06 Python
python相似模块用例
2016/03/04 Python
Tornado 多进程实现分析详解
2018/01/12 Python
pygame实现雷电游戏雏形开发
2018/11/20 Python
对Python+opencv将图片生成视频的实例详解
2019/01/08 Python
python爬虫 猫眼电影和电影天堂数据csv和mysql存储过程解析
2019/09/05 Python
Python如何读取、写入CSV数据
2020/07/28 Python
洛佩桑酒店官方网站:Lopesan Hotels
2019/04/15 全球购物
C#中的验证控件有几种
2014/03/08 面试题
简历中个人自我评价分享
2014/03/15 职场文书
新年主持词
2014/03/27 职场文书
政风行风评议工作总结
2014/10/21 职场文书
兵马俑的导游词
2015/02/02 职场文书
工作会议通知
2015/04/15 职场文书
麦田里的守望者读书笔记
2015/06/30 职场文书
任命书格式模板
2015/09/22 职场文书
《自己的花是让别人看的》教学反思
2016/02/19 职场文书
深入理解python多线程编程
2021/04/18 Python
python实现剪贴板的操作
2021/07/01 Python
JAVA 线程池(池化技术)的实现原理
2022/04/28 Java/Android