js函数的三种成创建方式以及它们各自的不同
js有三种创建函数的方式:
1.function语句(也叫函数声明)
function sum(a, b) {
return a + b;
}
sum(1, 2); // 3
2. 函数直接量,又叫函数字面量
它是一个表达式而不是语句。
var sum = function(a, b) {
return a + b;
}
sum(1, 2); // 3
上述代码就是函数通过直接量的方式进行定义。函数直接量是一个表达式,它可以定义匿名函数。函数直接量的语法和function语句十分相似,只不过, 它被用作表达式的一部分,而不是用作语句了。在函数直接量定义中,右侧的function所定义的是匿名函数,并将这个匿名函数的引用传递给表达式左侧。接下来执行了sum(1, 2),就运行了这个函数。
ps: 虽然函数字面量是一个匿名函数,但语法允许为其指定任意一个函数名,当写递归函数时可以调用它自己,使用Function()构造函数则不行。
var test = function factorial(x) {
if (x <= 1) {
return 1;
} else {
return x * factorial(x-1);
}
};
test(3); // 6
// 这个也是闭包的一种用法
3. 构造函数
它也是一个函数表达式。
var sum = new Function('a', 'b', 'return a + b;');
sum(1, 2); // 3
比较区别
首先我们来看看1、2之间的区别:
大家都知道有这样一个原理:函数声明格式
的函数存在函数声明提前,而函数表达式
、构造函数
都不存在函数声明提前。
所以咱们的js解析器就会先读取函数声明格式
的函数,让其在执行任何代码之前就可以访问;而函数表达式
则必须要等到js解析器执行到它所在的位置的时候才会被真正执行。
自执行函数
严格来说也叫函数表达式
,它主要用于创建一个新的作用域,在此作用域内声明的变量,不会和其他作用域内的变量冲突或混淆,大多以匿名函数方式存在,并且立即自动执行。
(function(a, b) {
console.log(a + b); // 3
})(1, 2)
var sum=(function(a, b){
return a + b;
})(1, 2);
console.log(sum); // 3
接下来我们来看第三种方式:
- 采用new Funciton的方式,这样会导致解析器会解析两次代码(第一次是解析常规ECMAScript代码,第二次则是解析传入构造函数中的字符串),从而影响性能。一般我们很少用到。
- 我们来看一个闭包的三种函数实现的栗子:
var a = 1;
function test1(){
var a = 2; // 局部变量
function test(){ return a; }; // function语句式,a=2
return test();
}
test1(); // 2
var a = 1;
function test1(){
var a = 2; // 局部变量
var test = function(){ return a;}; // 函数直接量,a=2 ,两者都是函数的作用域
return test();
}
test1(); // 2
var a = 1;
function test2(){
var a = 2; // 局部变量
var test = new Function("return a;"); // 构造函数,顶级的作用域,相当于在全局new一个函数,只能找到全局的a=1
return test();
}
test2(); // 1 没想到是1吧
用Function()构造函数创建一个函数时并不遵循典型的作用域,它一直把它当作是顶级函数来执行。顶级的作用域,相当于在全局new一个函数,只能找到全局的变量。
总结
最后用一张表格总结一下:
--- | function语句 | 函数直接量 | 构造函数 |
---|---|---|---|
创建形式 | function f(){} | var f = function(){} | var f = new Function() |
是否有名 | 有名字 | 匿名 | 匿名 |
性质 | 静态 | 静态 | 动态 |
解析机制 | 优先解析 | 顺序解析 | 顺序解析,且解析两次 |
性能 | 效率高 | 效率高 | 效率低 |
作用域 | 函数作用域 | 函数作用域 | 全局函数(具有顶级作用域),仅访问全局变量 |
动态和静态的区别
:function语句式 和 函数直接量 被javascript解释一次,放在了内存,以后用到的话直接调用,占用内存,所以说是“静态”的。而构造函数式是每一次调用都动态新建一个,用完之后就销毁了,不占用内存,所以说是“动态”的。所以,有些程序只希望调用一次,就可以用动态的构造函数式。
总结: js的三种创建方式都说完了,感觉讲还是比较清楚吧,参考学习了各方面的资料,这里加上自己的理解做了一次汇总。希望对自己也是一种提升。
转载自:
https://www.jscwwd.com/article/5e70976667f5cf03e3a5990b
更多前端文章在益码凭川博客网站
js函数的三种成创建方式以及它们各自的不同的更多相关文章
- js 函数定义三种方式
<p>Js 函数定义的三种方式:</p> <br> <p>方式一:function</p> <script type="te ...
- JS函数的三种方式
函数,一段能够自动完成某些功能的代码块,函数的出现,既解决了重复使用重一功能的需求,又可以避免代码的臃肿性. 使用函数有两个要求:必须调用后才可以执行;函数名不要和关键字以及系统函数相同; 函数主要有 ...
- 三种bean创建方式
- JavaScript创建函数的三种方式
㈠函数(function) ⑴函数也是一个对象 ⑵函数中可以封装一些功能(代码),在需要时可以执行这些功能(代码) ⑶函数中可以保存一些代码在需要的时候调用 ⑷使用typeof检查一个函数对象时,会返 ...
- js中的三种函数写法
js中的三种函数写法 <script type="text/javascript"> //普通的声明方式 function myFun(m,n){ alert(m+n) ...
- JS高级---三种创建对象的方式
JS高级---三种创建对象的方式 字面量的方式 (实例对象) 调用系统的构造函数 自定义构造函数方式 //创建对象---->实例化一个对象,的同时对属性进行初始化 var per=new Per ...
- 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之十二 || 三种跨域方式比较,DTOs(数据传输对象)初探
更新反馈 1.博友@落幕残情童鞋说到了,Nginx反向代理实现跨域,因为我目前还没有使用到,给忽略了,这次记录下,为下次补充.此坑已填 2.提示:跨域的姊妹篇——<三十三║ ⅖ 种方法实现完美跨 ...
- javascript中var let const三种变量声明方式
javascript中var let const三种变量声明方式 1.var ①var表示声明了一个变量,并且可以同时初始化该变量. ②使用var语句声明的变量的作用域是当前执行位置的上下文:一个函 ...
- Apache Spark探秘:三种分布式部署方式比较
转自:链接地址: http://dongxicheng.org/framework-on-yarn/apache-spark-comparing-three-deploying-ways/ 目 ...
随机推荐
- 阿里云系统安装部署Freeswitch
1.安装vim apt-get install vim 2.修改镜像源 将/etc/apt/source.list的原有源注释掉,添加下面的源: deb http://mirrors.163.com/ ...
- abstract class
在面向对象(OOP)语言中,一个类可以有一个或多个子类,而每个类都有至少一个公有方法作为外部代码访问的接口.而抽象方法就是为了方便继承而引入的,现在来看一下抽象类和抽象方法分别是如何定义以及他们的特点 ...
- Kinect_V1在Debian testing的配置指北
在Linux下驱动Kinect V1现在有两种方式,一种是使用OpenNI + SensorKinect + Nite的方案,一种是使用OpenNI2 + libfreenect的方案,第一种我没有尝 ...
- MSSS攝影大賽計劃書(第三版)
比賽內容:對香港的城市風景以及自然風光的攝影 預期成果: 提升同學對香港的認識,鼓勵學生走出大學學園去瞭解香港,同時豐富會員的課餘活動,培養同學的興趣愛好 比賽時間:4月1-15日 最後作品提交時間: ...
- MOOC(2)-Django开发get、post请求,返回json数据
1.对get请求直接返回参数 如果请求多个参数,也只能返回一个参数,这里只返回了username参数 如果想要返回多个参数值,可以返回json格式数据 2.对get请求返回json数据 # views ...
- springboot中使用异步的常用两种方式及其比较
一般对于业务复杂的流程,会有一些处理逻辑不需要及时返回,甚至不需要返回值,但是如果充斥在主流程中,占用大量时间来处理,就可以通过异步的方式来优化.实现异步的常用方法远不止两种,但是个人经验常用的,好用 ...
- dubbo分布式框架下web层调用业务层一直报空指针异常的解决办法
java.lang.NullPointerException............... 环境:SSM(通用mapper)+Dubbo 1.检查导包 提示注解@Reference 应该导入 im ...
- cesium入门示例-3dTiles加载
数据转换工具采用cesiumlab1.5.17版本,转换后的3dTiles加载显示比较简单,通过Cesium.Cesium3DTileset接口指定url即可,3dTiles文件可与js前端代码放置一 ...
- drf三大认证
源码分析 """ 1)APIView的dispath(self, request, *args, **kwargs) 2)dispath方法内 self.initial( ...
- 吴裕雄--天生自然python学习笔记:beautifulsoup库的使用
Beautiful Soup 库简介 Beautiful Soup提供一些简单的.python式的函数用来处理导航.搜索.修改分析树等功能.它是一个工具箱,通过解析文档为用户提供需要抓取的数据,因为简 ...