ES6之函数的扩展

1. ES6之函数的扩展

1.1. 函数参数的默认值

ES6 之前,不能直接为函数的参数指定默认值,只能采用变通的方法。

function log(x, y) {
  y = y || 'World';
  console.log(x, y);
}

log('Hello') // Hello World
log('Hello','China') // Hello China
log('Hello','') // Hello World

上面代码检查函数 log 的参数 y 有没有赋值,如果没有,则指定默认值为 World 。

这种写法的缺点在于,如果参数 y 赋值了,但是对应的布尔值为 false ,则该赋值不起作用。

就像上面代码的最后一行,参数 y 等于空字符,结果被改为默认值。

ES6 允许为函数的参数设置默认值,即直接写在参数定义的后面。

function log(x, y = 'World') {
  console.log(x, y);
}

log('Hello') // Hello World
log('Hello','China') // Hello China
log('Hello','') // Hello
1.1.1. 注意事项
1.1.1.1. 参数变量是默认声明的,所以不能用 let 或 const 再次声明。
function foo(x = 5) {
  let x = 1; // error
  const x = 2; // error
}
1.1.1.2. 使用参数默认值时,函数不能有同名参数。
// 不报错
function foo(x, x, y) {
  // ...
}
// 报错
function foo(x, x, y = 1) {
  // ...
}
// SyntaxError: Duplicate parameter name not allowed in this context
1.1.1.3. 参数默认值是惰性求值的。

参数默认值不是传值的,而是每次都重新计算默认值表达式的值。

let x = 99;
function foo(p = x + 1) {
  console.log(p);
}

foo() // 100

x = 100;
foo() // 101

上面代码中,参数 p 的默认值是 x + 1 。这时,每次调用函数 foo ,都会重新计算 x + 1 ,而不是默认 p 等于 100。

1.1.2. 与解构赋值默认值结合使用

参数默认值可以与解构赋值的默认值,结合起来使用。

function foo({x, y = 5} = {}) {
  console.log(x, y);
}

foo() // undefined 5

双重默认值

function fetch(url, 
{ 
  body = ''
  , method = 'GET'
  , headers
  = {} 
} = {}) {
  console.log(method);
}

fetch('http://example.com')
// "GET"

1.2. 函数的 length 属性

指定了默认值以后,函数的 length 属性,将返回没有指定默认值的参数个数

也就是说,指定了默认值后, length 属性将失真。

(function (a) {}).length // 1
(function (a = 5) {}).length // 0
(function (a, b, c = 5) {}).length // 2

如果设置了默认值的参数不是尾参数,那么length 属性也 不再计入后面的参数了

(function (a = 0, b, c) {}).length // 0
(function (a, b = 1, c) {}).length // 1

1.3. rest 参数

ES6 引入 rest 参数(形式为 …变量名 ),用于获取函数的多余参数,这样就不需要使用 arguments 对象了。

rest 参数搭配的变量是一个数组,该变量将多余的参数放入数组中。

function add(...values) {
  let sum = 0;
  for (var val of values) {
    sum += val;
  }
  return sum;
}
add(2, 5, 3) // 10

上面代码的 add 函数是一个求和函数,利用 rest 参数,可以向该函数传入任意数目的参数。

arguments 对象不是数组,而是一个类似数组的对象。所以为了使用数组的方法,必须使用 Array.prototype.slice.call 先将其转为数组。

rest 参数就不存在这个问题,它就是一个真正的数组,数组特有的方法都可以使用。

下面是一个利用 rest 参数改写数组 push 方法的例子。

function push(array, ...items) {
  items.forEach(function(item) {
    array.push(item);
    console.log(item);
  });
}

var a = [];
push(a, 1, 2, 3)
1.3.1. 注意事项
  • rest 参数之后不能再有其他参数(即只能是最后一个参数),否则会报错。
  • 函数的 length 属性,不包括 rest 参数。

1.4. 严格模式

从 ES5 开始,函数内部可以设定为严格模式。

function doSomething(a, b) {
  'use strict';
  // code
}

ES2016 规定只要函数参数使用了默认值、解构赋值、或者扩展运算符,那么函数内部就不能显式设定为严格模式,否则会报错。

// 报错
function doSomething(a, b = a) {
  'use strict';
  // code
}
// 报错
const doSomething = function ({a, b}) {
  'use strict';
  // code
};
// 报错
const doSomething = (...a) => {
  'use strict';
  // code
};
const obj = {
  // 报错
  doSomething({a, b}) {
    'use strict';
    // code
  }
};

1.5. 箭头函数

ES6 允许使用“箭头”( => )定义函数。

var f = v => v;

上面的箭头函数等同于:

var f = function(v) {
  return v;
};

如果箭头函数不需要参数或需要多个参数,就使用一个圆括号代表参数
部分。

var f = () => 5;
// 等同于
var f = function () { return 5 };


var sum = (num1, num2) => num1 + num2;
// 等同于
var sum = function(num1, num2) {
  return num1 + num2;
};
1.5.1. 箭头函数有几个使用注意点。
  • (1)函数体内的 this 对象,就是定义时所在的对象,而不是使用时所在的对象。
  • (2)不可以当作构造函数,也就是说,不可以使用 new 命令,否则会抛出一个错误。
  • (3)不可以使用 arguments 对象,该对象在函数体内不存在。如果要用,可以用 rest 参数代替。
  • (4)不可以使用 yield 命令,因此箭头函数不能用作 Generator 函数。

2. 总结

ES6(ECMAScript 2015)对函数进行了多方面的扩展,旨在提高代码的简洁性、可读性和功能性。

以下是ES6中函数扩展的一些关键特性:

2.1. 默认参数(Default Parameters):

允许为函数参数提供默认值,如果调用函数时没有传入对应的参数值,就会使用默认值。这消除了在函数体内部进行参数检查的需要。

2.2. 箭头函数(Arrow Functions):

引入了一种新的函数表达式语法,使用=>符号定义。箭头函数不仅更简洁,还自动绑定词法作用域的this值,解决了传统函数中this指向的常见问题。

2.3. 函数参数的解构赋值(Destructuring in Parameter Lists):

在函数参数列表中可以直接解构数组或对象,使得处理复杂参数更加方便。同时,解构的参数也可以有默认值。

2.4. Rest 参数(Rest Parameters):

使用...操作符,允许你将一个不确定数量的参数作为一个数组传递给函数,替代了传统的arguments对象,并且具有更清晰的语义。

2.5. Spread 操作符在函数调用中的应用

同样使用...操作符,可以在调用函数时将数组或类数组对象的元素展开为独立的参数。

2.6. 函数名称属性(name Property):

所有函数现在都有一个name属性,可以更方便地获取函数的名字,即使是匿名函数或通过某些构造方式创建的函数。

2.7. 块级作用域中的函数声明

在ES6的块级作用域(如if语句或for循环内)中声明的函数不会被提升到全局或外部作用域,它们只在声明它们的块中可见。

2.8. 参数默认值的位置约束

如果有参数设置了默认值,这些参数必须放在没有默认值的参数之后。

2.9. 函数的length属性变化

如果函数参数设置了默认值,那么length属性将不包括有默认值的参数。

这些扩展大大丰富了JavaScript函数的表达能力和灵活性,使得编写现代JavaScript代码更加高效和愉悦。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

前端布道人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值