1.函数参数的默认值
Ⅰ.基本结构
- 为函数的参数设置默认值
- 参数变量是默认声明的不能用let或const再次声明 →报错
- 不能有同名参数
- 默认值每一次使用都会重新计→惰性求值
let x=9;
function foo(p = x + 1){
console.log(p)
}
foo(); //10
x=10;
foo(); //11
Ⅱ.与解构赋值默认值结合使用
function foo({x,y = 5}){
console.log(x,y);
}
foo({}); //undefined 5
foo({x:1}); //1,5
foo({x:1,y:2)}; //1,2
foo(); //报错
- 使用了对象的解构赋值默认值。只有参数是一个对象时,变量才会通过解构赋值。→ 函数没有提供参数,变量 x,y你会生成,报错。
- 避免 → 通过提供函数参数的默认值;
没有提供参数,参数默认为一个空对象undefined
function foo({ x , y = 5} = { }){
console.log(x,y);
}
foo(); //undefined 5
- 两种写法的区别?
function m1({x : 0, y = 0} = {})
function m2({x,y} = {x : 0,y : 0})
m1➜参数默认值是空对象,设置了解构赋值的默认值
m2➜函数默认值是一个有具体属性的对象,没有设置对象解构赋值的默认值
Ⅲ.参数默认值位置
- 通常情况下,默认值参数 → 尾参数
- 如果默认值的尾参数,则无法省略该参数,而不省略它后面的参数,除非显示输入 undefined (传入undefined ,将触发参数的默认值)
function f(x=1,y){}
f(,1); ✘
f(undefined,1); ✔
Ⅳ.函数的lenght属性
- 对 length 指定默认值,length 属性失真(将返回没有指定默认值的参数个数????)
- length返回值 = 函数参数个数 - 指定默认值的参数个数
- rest 参数不计入length 属性
Ⅴ.作用域
- 设置函数的默认值+函数声明初始化 ➔ 参数形成单独的作用域
初始化结束后 ➔ 作用域消失
以上现象在不设置参数默认值时,不出现
Ⅵ.应用
- 可以指定某一个参数不得省略
- 参数值设hi为 undefined,表明参数可以省略
2.rest 参数
- 形式 :rest 参数(...变量名) ➔ 数组,将多余的变量放进数组中
用于会获取函数的多余参数,可以不使用arguments对象了 - 对比:arguments 对象不是数组,而是一个类似数组的对象
- 注意: rest参数只能是最后一个参数,否则会报错
- 函数的length属性,不包括reat参数
3.严格模式
- 函数内部可以设定为严格模式
'use strict' ➔ ES6规定,函数参数使用默认值,解构赋值,扩展运算符,函数内部不能显示为严格模式,否则报错 - 严格模式下,不能使用前缀0表示八进制 (Js引擎会先成功执行,进入函数内部,发现是严格模式,然后报错)
- 解决限制:
①设定全局性的严格模式,合法
②把函数包在一个无参数的立即执行函数
4.name属性
- 返回该函数的函数名
- ES6对属性行为做出了一些修改:
匿名函数: ES5 的name属性,会返回空字符串
ES6的name属性,返回世纪的函数名
var f = function(){}
//ES5
f.name // ""
//ES6
f.name // "f"
- 将一个具名函数赋值给一个变量,ES5/6的name属性都返回这个具名函数原本的名字
const bar = function baz(){ };
//ES5
bar.name // "baz"
//ES6
bar.name //"baz"
- Founction 构造函数返回的函数实例,name 的属性值为 anonymous
(new Function).name
→ 'anonyous'
- bind 返回的函数,name属性值前会加上 bound
4.箭头函数 ( => )
- 关于参数
- var f = v => v ; 单个参数
- var f = ( a, b) => a + b; 多个参数用括号包起来
- 箭头函数代码块多于一条语句,使用 {}
- 返回一个对象,必须在对象外面加上括号
- 只有一行语句,不需要返回值
let fn = () => void doesNotReturn();
- 与变量解构结合使用
- 简化回调函数
- 与 rest 参数结合
使用注意
❶ 函数体内的 this 对象,是定义时所在的对象,不是使用时所在的对象
❷ 不可以当作构造函数,否则报错
❸ 不可使用 arguments 对象,函数内不存在(rest 参数代替)
❹ 不可使用 yield 命令,则不能做 Generator 函数
不适用场合
❶ 定义对象的方法
❷ 使用动态 this 的时候(箭头函数将 this 从 动态 装换成 静态)
5.尾调用优化
只在严格模式下开启,正常模式无效
尾调用指某个函数的最后一步是调用另一个函数(不一定出现在函数的尾部,主要看是否是操作的最后一步)
尾调用优化 只保留内层函数的调用帧。节省内存
函数调用会在内存中形成一个“调用记录”➨调用帧,保存调用位置和内部变量等信息。
尾调用由于是函数的最后一步操作,所以不需要保留外层函数的调用帧,因为调用位置、内部变量等信息都不会再用到,只要直接用内层函数的调用帧,取代外层函数的调用帧就可以。
尾递归
▼
尾调用自身
优点 普通递归耗费内存,容易发生“栈溢出”错误,尾递归,只存在一个调用帧,则永远不会出现“栈溢出”
柯里化
将多参数的函数装换成单参数的形式