Javascript高级编程学习笔记(23)—— 函数表达式(1)递归

前面的文章中,我在介绍JS中引用类型的时候提过,JS中函数有两种定义方式

第一种是声明函数,即使用function关键字来声明

第二种就是使用函数表达式,将函数以表达式的形式赋值给一个变量,这个变量就保存了对这个函数的引用

function与表达式的区别前文也已经详细分析过,这里就大概讲解一下

虽然两者在写法上没有什么不同,但是function关键字作为函数声明的时候,与var一样会有变量提升的效果

其本质是在JS创建执行环境的时候,就完成了对函数的声明,所以访问该函数的代码在函数声明语句前就可以使用

但是函数表达式不一样,在JS中 = 后面的内容会被归为表达式上下文,而这种上下文中 function 关键字并没有变量提升的特权

因为,这段表达式的代码,会在执行上下文创建完成后才,会被执行

函数表达式定义函数

var a = function(){
// 代码块
}

上面这种定义函数的方式,也是我们最常用的函数表达式的定义方式

我们在这种情况下没必要给函数一个名字,因为已经有一个变量保存了对该函数的引用。

和函数名的作用一样,所以没必要重复创建

PS.这种没有标识符的函数叫匿名函数,也叫拉姆达函数,函数的name属性为空字符串

在流程控制语句中使用函数表达式

可能许多小伙伴听过一个说法,那就是不要在 if for while 中声明函数

这是为什么呢?

首先我们应该明确,这里指的声明是指使用 function 关键字进行函数声明,而不是使用函数表达式

不让大家在流程控制语句中进行函数声明的主要原因就是,这种写法虽然在逻辑上没有问题,但是在JS解析时会出现错误

我们刚才说过,使用function 关键字声明的函数会有个变量提升的过程

在创建执行上下文的时候,函数的定义就已经完成了

那么问题来了

流程控制语句在代码执行时才会进行判断,而你的声明在判断之前就已经完成了

所以在大部分浏览器中你的流程控制语句并不会有任何作用

但在小部分浏览器,比如火狐针对这一问题是可以按照逻辑完成声明的

所以我们不应该在流程控制语句中进行函数声明

但是话又说回来,万一我就是要在流程控制语句中按照情况来定义函数呢?

那么函数表达式就是你的理想选择

因为不让用的原因就是 function 的变量提升,而函数表达式又不会变量提升,当然就可以安全地使用了

除此而外,当我们需要将函数作为值传递的时候,函数表达式依然是你的首选(避免一些奇奇怪怪的错误)

递归

我的这系列文章,主要是记录自己在学习JS中的一些知识点

所以并不会在这里讲递归算法

主要是说一下递归中存在的问题

抛开ES6中的尾调用优化不谈

在ES5中,我们平时习以为常的递归写法很可能存在一些隐患

function factorial(num){
if(num <= 1){
return 1;
}else{
return num * factorial(num-1);
}
}

上方是一个经典的阶乘递归函数

小伙伴们觉得有啥问题么?

没看出来,没关系

看看下面的代码

var anotherFactorial = factorial;
factorial = null;
anohterFactorial(5); // 报错

我先用一个变量来保存对这个递归函数的引用

然后去掉它原本的引用

但我通过新的引用执行函数时,js报错了

为啥?

因为我们的递归使用的是原来的引用来进行递归调用的

而我们已经切断了原来的引用,所以函数执行时报错

换句话说,这种递归函数的写法耦合度较高

那么怎么解决呢?

function factorial(num){
if(num <= 1){
return 1;
}else{
return num * arguments.callee(num-1);
}
}

之前我们说过,arguments有个callee属性指向函数本身

这里用这种方法就可以解决之前的问题

但是这种方法还有一个问题,在严格模式下不可用

怎么办呢?

var factorial = function f(num){
if(num <= 1){
return 1;
}else{
return num * f(num-1);
}
};

当然是使用函数表达式啦,这种方式在严格模式下依然可用

以上就是函数表达式关于递归的内容啦,明天更闭包的内容,不见不散Javascript高级编程学习笔记(23)——  函数表达式(1)递归

上一篇:OC对象:封装、继承、多态的使用举例一


下一篇:Greedy:三角形问题