本文主要是介绍【亡羊补牢】JS灵魂之问 第14期 修炼内功 函数基础 形参实参映射关系,是什么?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
引言
有几天没有更新JS灵魂之问的专栏系列文章了,秋招季,也在忙着备战笔试面试。今天得空再来写一篇文章,本篇要讲解的内容是关于 函数基础
,那今天这篇看能不能问倒你了,一起来探索一下吧。
仰望星空的人,不应该被嘲笑
文章目录
- 引言
- 函数基础
- 函数种类、字面量
- 形参实参映射
- 作用域引入
- 函数默认参数
- 最后
函数基础
函数种类、字面量
开门见山,下面代码会输出什么,不知道小伙伴平常有咩有想过这类问题
var test = function test1() {var a = 1;console.log(a);
}
test()
test1()
答案是 1
和 报错
,报错信息如下:
这是为什么呢?实际上,我们申明了一个变量 test
,把这个函数 test1
赋给它的时候,就已经把这个变量赋予了函数的功能,调用 test()
方法当然能够正常运行,输出 1
。而对于表达式赋值,会自动忽略后面的函数名称,也就是说写与不写并不影响 test()
方法的执行。不信,看看下面代码会输出什么?
var test = function() {var a = 1;console.log(a);
}
test() // 1
那这样,是不是说 test1
完全没作用了,那写着干嘛,还多几个字符。当然不是!,test1
在函数体内部是可见的,而在外部却不可见,通过这样,我们就可以实现递归操作
var test = function() {var a = 1;console.log(a);
}
而对于上述代码,后面函数体没有名字,我们称之为 匿名函数,是不是有点印象了,原来就是这东西,哟西~
而通过这种方式赋值的表达式,我们称之为 匿名函数表达式,也称为 函数字面量,这些专有名词一出来,瞬间觉得有点逼格了有没有!
字面量这种东西,简单来说就是数据,例如下述,有数字字面量,字符串字面量,数组字面量
等等。简单理解就是,对于赋值的过程,右边的数据就是字面量。
var a = 10
var b = '111'
var c = [1,2,3]
形参实参映射
补充:如何获取形参和实参对应的长度?
function test (a,b) {console.log(test.length) // 形参的长度 2console.log(arguments.length) // 实参的长度 3
}
test(1,2,3)
我想小伙伴们应该清楚实参和形参是什么玩意,但是我们可以更改实参的值吗?例如下述代码,会输出什么呢?
function test(a, b) {a =3;console.log(arguments[0]);
}
test(1,2);
答案是 3
,我们可以修改实参的值。
刚刚那题只是简单热个身,继续下一题吧,我们可以改变 b
的值吗?(提示:注意我并没有传对应实参哦~)
function test(a, b) {b = 3;console.log(arguments[1]);
}
test(1);
答案是 undefined
,因此对于上一题表述,要修改一下:对于实参传递过来确定的值,我们是可以进行修改的,而如果实参并没有传递值过来,我们是不能进行修改的。这就是形参和实参的映射关系。
简单解释一下形参和实参的映射关系,其实实参和形参不能说是一类的,看上述代码,我们可以通过 arguments
来获取我们的实参,可以看做是一个数组里面的某一项值,而数组是存放堆内存的,而对应我们形参其实是存放在栈内存的,它们之间会有一个映射关系,并且是一对一对应的,上述我们实参没有对b
进行赋值,尽管修改了形参,但改变不了我们的 arguments[1]
就是这个道理。(没有建立一对一映射关系)。
作用域引入
再来一道引申题,为后续内容做铺垫。下面 a b c
分别会输出什么?
a = 1;
function test1 () {var b = 2;console.log(a) // function test2() {var c = 3console.log(b); // }test2();console.log(c); //
}
test1();
答案是 1 2 报错
,这就牵扯到 scope
问题了,简单理解就是函数内部能访问外面的变量,而函数外面却不能访问内部的变量,也就是闭包问题。(这个后文会提到)
函数默认参数
如果实参没有赋值,那么形参怎样设置默认参数呢?说到默认参数,我想你应该会想到如下代码:
function test(a = 1, b = 1) {console.log(a)console.log(b)
}
test() // 1 1
好的,上述问题算是开胃小菜,我们继续,我如果给 a
设定默认值,而 b
通过实参传递过来呢?可以实现吗?之前没有传参的话,不是默认打印 undefined
吗,那我现在给 a
传递一个 undefined
,是不是就会定为默认值。
function test(a = 1, b) {console.log(a)console.log(b)
}
test(undefined, 2)
答案是可以的,上述代码输出结果为 1 2
。简单解释一下,在之前我们将了形参实参是有一个映射关系,对于堆内存 arguments
里面,如果给了 undefined
,那么就会去栈内存形参里面找,如果不为 undefined
,则会设置形参的默认值。(其实这是 es6
的语法了)
那么,可以用es5
的方式实现一下吗?(当然可以,见代码)
function test(a, b) {a = arguments[0] || 1b = arguments[1] || 1console.log(a)console.log(b)
}
test(undefined, 2)
最后
文章产出不易,还望各位小伙伴们支持一波!
往期精选:
小狮子前端の笔记仓库
访问超逸の博客,方便小伙伴阅读玩耍~
学如逆水行舟,不进则退
这篇关于【亡羊补牢】JS灵魂之问 第14期 修炼内功 函数基础 形参实参映射关系,是什么?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!