# 语法
~~~js
function(a, b, ...theArgs) {
// ...
}
~~~
# 描述
如果函数的最后一个命名参数以`...`为前缀,则它将成为一个由剩余参数组成的真数组,其中从`0`(包括)到`theArgs.length`(排除)的元素由传递给函数的实际参数提供。
在上面的例子中,`theArgs`将收集该函数的第三个参数(因为第一个参数被映射到`a`,而第二个参数映射到`b`)和所有后续参数。
## 剩余参数和 `arguments`对象的区别
剩余参数和 `[arguments]`对象之间的区别主要有三个:
* 剩余参数只包含那些没有对应形参的实参,而`arguments`对象包含了传给函数的所有实参。
* `arguments`对象不是一个真正的数组,而剩余参数是真正的 `[Array]`实例,也就是说你能够在它上面直接使用所有的数组方法,比如 `[sort]`,`[map]`,`[forEach]`或`[pop]`。
* `arguments`对象还有一些附加的属性 (如`callee`属性)。
## 从 arguments 到数组
引入了剩余参数来减少由参数引起的样板代码。
~~~js
// Before rest parameters, "arguments" could be converted to a normal array using:
function f(a, b) {
var normalArray = Array.prototype.slice.call(arguments);
// -- or --
var normalArray = [].slice.call(arguments);
// -- or --
var normalArray = Array.from(arguments);
var first = normalArray.shift(); // OK, gives the first argument
var first = arguments.shift(); // ERROR (arguments is not a normal array)
}
// Now we can easily gain access to a normal array using a rest parameter
function f(...args) {
var normalArray = args;
var first = normalArray.shift(); // OK, gives the first argument
}
~~~
> 您还可以使用[`Array.from()`](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/from "Array.from() 方法从一个类似数组或可迭代对象中创建一个新的,浅拷贝的数组实例。")方法或[扩展运算符](https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Operators/Spread_operator)将参数转换为真实数组:
~~~js
var args = Array.from(arguments);
var args = [...arguments];
~~~
## 解构剩余参数
剩余参数可以被解构,这意味着他们的数据可以被解包到不同的变量中。
~~~js
function f(...[a, b, c]) {
return a + b + c;
}
f(1) // NaN (b and c are undefined)
f(1, 2, 3) // 6
f(1, 2, 3, 4) // 6 (the fourth parameter is not destructured)
~~~
## 示例
因为`theArgs`是个数组,所以你可以使用`length`属性得到剩余参数的个数:
~~~js
function fun1(...theArgs) {
alert(theArgs.length);
}
fun1(); // 弹出 "0", 因为theArgs没有元素
fun1(5); // 弹出 "1", 因为theArgs只有一个元素
fun1(5, 6, 7); // 弹出 "3", 因为theArgs有三个元素
~~~
下例中,剩余参数包含了从第二个到最后的所有实参,然后用第一个实参依次乘以它们:
~~~js
function multiply(multiplier, ...theArgs) {
return theArgs.map(function (element) {
return multiplier * element;
});
}
var arr = multiply(2, 1, 2, 3);
console.log(arr); // [2, 4, 6]
~~~
下例演示了你可以在剩余参数上使用任意的数组方法,而`arguments`对象不可以:
~~~js
function sortRestArgs(...theArgs) {
var sortedArgs = theArgs.sort();
return sortedArgs;
}
alert(sortRestArgs(5,3,7,1)); // 弹出 1,3,5,7
function sortArguments() {
var sortedArgs = arguments.sort();
return sortedArgs; // 不会执行到这里
}
alert(sortArguments(5,3,7,1)); // 抛出TypeError异常:arguments.sort is not a function
~~~
为了在`arguments`对象上使用`Array`方法,它必须首先被转换为一个真正的数组。
~~~js
function sortArguments() {
var args = Array.prototype.slice.call(arguments);
var sortedArgs = args.sort();
return sortedArgs;
}
console.log(sortArguments(5, 3, 7, 1)); // shows 1, 3, 5, 7
~~~
# 实践
## 遍历参数求和
### 使用剩余参数和高阶函数reduce
```js
function sum(...theArgs) {
return theArgs.reduce((previous, current) => {
return previous + current;
});
}
console.log(sum(1, 2, 3));
// expected output: 6
console.log(sum(1, 2, 3, 4));
// expected output: 10
```
### 使用arguments
~~~js
function sum() {
var sigma=0,
len = arguments.length;
for(var i=0; i<len; i++){
sigma+= arguments[i];
}
return sigma;
}
sum() // 0
sum(1) // 1
sum(1,2,3,4); // 10
~~~
- 内容介绍
- EcmaScript基础
- 快速入门
- 常量与变量
- 字符串
- 函数的基本概念
- 条件判断
- 数组
- 循环
- while循环
- for循环
- 函数基础
- 对象
- 对象的方法
- 函数
- 变量作用域
- 箭头函数
- 闭包
- 高阶函数
- map/reduce
- filter
- sort
- Promise
- 基本对象
- Arguments 对象
- 剩余参数
- Map和Set
- Json基础
- RegExp
- Date
- async
- callback
- promise基础
- promise-api
- promise链
- async-await
- 项目实践
- 标签系统
- 远程API请求
- 面向对象编程
- 创建对象
- 原型继承
- 项目实践
- Classes
- 构造函数
- extends
- static
- 项目实践
- 模块
- import
- export
- 项目实践
- 第三方扩展库
- immutable
- Vue快速入门
- 理解MVVM
- Vue中的MVVM模型
- Webpack+Vue快速入门
- 模板语法
- 计算属性和侦听器
- Class 与 Style 绑定
- 条件渲染
- 列表渲染
- 事件处理
- 表单输入绑定
- 组件基础
- 组件注册
- Prop
- 自定义事件
- 插槽
- 混入
- 过滤器
- 项目实践
- 标签编辑
- iView
- iView快速入门
- 课程讲座
- 环境配置
- 第3周 Javascript快速入门