【译】ECMAScript 6中的参数使用
ECMAScript 6(又名 ECMAScript2015)是 ECMAScript 标准的最新版本,在 JavaScript 中明显提升了参数的使用。现在我们可以使用 rest 参数、默认值和解构等其它特性。
在本教程中,我们将探讨实参和形参的详情,看看 ECMAScript 6 对其升级了哪些特性。
# Arguments 与 Parameters
Arguments 和 parameters 通常是相互使用。然而在本教程中我们将做一个区分。通常 ,parameters(形参)是在函数声明时指定的,而 arguments(实参)是调用函数时传入的 。思考下面这个函数:
function foo(param1, param2) {
// do something
}
foo(10, 20);
2
3
4
在这个函数中,param1
和param2
是函数的形参,传入函数的值(10
和20
)是实参。
# 扩展运算符(...)
在 ECMAScript 5 中,apply()
方法传递一个数组作为函数的参数是方便的。例如,常
用Math.max()
方法去查找一个数组中的最大值,思考下面这段代码:
var myArray = [5, 10, 50];
Math.max(myArray); // Error: NaN
Math.max.apply(Math, myArray); //50
2
3
Math.max()
方法不支持数组参数,它仅仅接受数值类型的参数列表。当一个数组传入
到Math.max()
函数中时,它抛出一个错误。但是当使用apply()
方法后,数组会分解为
单独的数字传入到Math.max()
方法中,所以Math.max()
方法可以处理它。
幸运的是,随着在 ECMAScript 6 扩展操作符的引入,我们不再需要使用apply()
方法。
使用扩展操作符,我们可以轻易地将一个表达式扩展为多个参数:
var myArray = [5, 10, 50];
Math.max(...myArray); // 50
2
在这里,扩展运算符将myArray
展开成一个个单独的参数。在 ECMAScript 5 中使
用apply()
模拟扩展运算符是可行的,但是语法是混乱的,并且缺乏扩展运算符的灵活性
。扩展运算符不但使用简单而且还具有更多的特性。例如,在函数调用时,它可以多次使用
并且可以和其它参数混合使用:
function myFunction() {
for (var i in arguments) {
console.log(arguments[i]);
}
}
var params = [10, 15];
myFunction(5, ...params, 20, ...[25]); // 5 10 15 20 25
2
3
4
5
6
7
扩展操作符的另一个优点是,它可以很容易地与构造函数一起使用:
new Date(...[2016, 5, 6]); // Mon Jun 06 2016 00:00:00 GMT-0700 (Pacific Daylight Time)
当然,我们可以在 ECMAScript 5 中重写上面的代码,但我们需要使用复杂的模式来避免类 型错误:
new Date.apply(null, [2016, 4, 24]); // TypeError: Date.apply is not a constructor
new (Function.prototype.bind.apply(Date, [null].concat([2016, 5, 6])))(); // Mon Jun 06 2016 00:00:00 GMT-0700 (Pacific Daylight Time)
2
# 浏览器对扩展操作符的支持
桌面浏览器 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE | OPERA |
SAFARI |
---|
46 |
MOBILE |
---------------------- |
46 |
# Rest 参数
rest 参数的语法与扩展运算符一样,但是它不是展开一个数组到参数中,相反它是将参数 收集为一个数组:
function myFunction(...options) {
return options;
}
myFunction('a', 'b', 'c'); //["a", "b", "c"]
2
3
4
如果没有实参,rest 参数将被设置为一个空数组:
function myFunction(...options) {
return options;
}
myFunction(); // []
2
3
4
当创建一个参数可变的函数(即接受的参数数量是不确定的函数)时,rest 参数是非常有
用的。得益于数组的好处,reset 参数可以容易地替换掉arguments
对象(我们将在本教
程后面解释)。思考这个使用 ECMAScript 5 写的函数:
function checkSubstring(string) {
for (var i = 1; i < arguments.length; i++) {
if (string.indexOf(arguments[i]) === -1) {
return false;
}
}
return true;
}
checkSubstring('this is a string', 'is', 'this'); // true
2
3
4
5
6
7
8
9
这个函数检查一个字符串是否包含子字符串。这个函数的第一个问题是,我们必须得看函数
的内部才知道它需要多个参数。第二个问题是循环迭代必须从索引1
而不是0
开始,因
为arguments[0]
指向第一个参数。如果我们以后决定在string
参数之前或者之后添加另
一个参数,我们可能会忘记更新 for 循环的逻辑处理。使用 rest 参数,我们就很容易地
避免这些问题:
function checkSubstrings(string, ...keys) {
for (var key of keys) {
if (string.indexOf(key) === -1) {
return false;
}
}
return true;
}
checkSubstrings('this is a string', 'is', 'this'); // true
2
3
4
5
6
7
8
9
此函数的输出和前一个函数相同。在这里,string
参数是传入的第一个参数,其它的参数
放进一个数组中,并赋值给了变量keys
。
使用 rest 参数代替arguments
对象,提高了代码的可读性,避免了 JavaScript 中
的优化问题 (opens new window)。
然而,rest 参数也不是没有它的局限性。例如,它必须是最后一个参数,否则,将出现语
法错误。
function logArguments(a, ...params, b) {
console.log(a, params, b);
}
logArguments(5, 10, 15); // SyntaxError: parameter after rest parameter
2
3
4
另一个限制是,在函数声明中只允许存在一个 rest 参数:
function logArguments(...param1, ...param2) {
}
logArguments(5, 10, 15); // SyntaxError: parameter after rest parameter
2
3
# 浏览器对 rest 参数的支持
桌面浏览器 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE | OPERA |
SAFARI |
---|
47 |
MOBILE |
---------------------- |
47 |
# 默认参数
# ECMAScript 5 中的默认参数
在 ECMAScript 5 中 JavaScript 不支持默认参数,但有一个简单的解决方法。在函数内使
用逻辑or
操作符(||
),我们很容易在 ECMAScript 5 中模拟默认参数。思考这个函数:
function foo(param1, param2) {
param1 = param1 || 10;
param2 = param2 || 10;
console.log(param1, param2);
}
foo(5, 5); // 5 5
foo(5); // 5 10
foo(); // 10 10
2
3
4
5
6
7
8
这个函数预计需要有两个参数,但是当它不带参数调用时,它将使用默认值。在函数内部,
缺少的参数会自动设置为 undefined。因此,我们可以检测到这些参数,并为它们声明为默
认值。检测缺失的参数和设置默认值,我们使用逻辑or
操作符(||
)。这个操作符检查
表达式的第一个值:如果它
是真值(truthy) (opens new window),操
作符返回它。如果它是假值,操作符返回第二个值。
我们经常在函数中使用这种方式来设置参数的默认值,但是它是有缺陷的。向函数传
入0
和null
也会触发默认值,因为这些参数的值会被转换
为假值(falsy) (opens new window)。因此
,如果我们真的需要给函数传入0
和null
,我们需要另一种方法来检查是否缺少参数:
function foo(param1, param2) {
if (param1 === undefined) {
param1 = 10;
}
if (param2 === undefined) {
param2 = 10;
}
console.log(param1, param2);
}
foo(0, null); // 0, null
foo(); // 10, 10
2
3
4
5
6
7
8
9
10
11
在这个函数中,对传入的参数进行类型检查,以确保在赋默认值之前,它们是 undefined。
这种方式需要多写一点的代码,但它是一个更安全的选择,并且允许我们向函数传入参
数0
和null
。
# ECMAScript 6 中的默认参数
使用 ECMAScript 6,我们不再需要检查 undefined 值来模拟默认参数了,现在我们可以将 默认值放在函数声明中:
function foo(a = 10, b = 10) {
console.log(a, b);
}
foo(5); // 5 10
foo(0, null); // 0 null
2
3
4
5
正如你所看到的,调用函数时,省略参数会触发默认值,但是传入0
和null
并不会触发
默认值。我们甚至可以使用函数返回值作为参数的默认值:
function getParam() {
alert('getParam was called');
return 3;
}
function multiply(param1, param2 = getParam()) {
return param1 * param2;
}
multiply(2, 5); // 10
multiply(2); // 6 (also displays an alert dialog)
2
3
4
5
6
7
8
9
注意,函数getParam
仅仅在第二个参数缺省时才会被调用。因此当我们传入两个参数调用
函数multiply()
时,alert 并不会显示。
默认参数另一个有趣的特性是,在函数声明中我们可以引用其他参数和变量:
function myFunction(a = 10, b = a) {
console.log('a = ' + a + '; b = ' + b);
}
myFunction(); // a=10; b=10
myFunction(22); // a=22; b=22
myFunction(2, 4); // a=2; b=4
2
3
4
5
6
你甚至可以在函数声明中执行操作:
function myFunction(a, b = ++a, c = a * b) {
console.log(c);
}
myFunction(5); // 36
2
3
4
注意,与其它语言不同,JavaScript 是在调用函数时计算默认值。
function add(value, array = []) {
array.push(value);
return array;
}
add(5); // [5]
add(6); // [6], not [5, 6]
2
3
4
5
6
# 浏览器对默认参数的支持
桌面浏览器 | 特性 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE |
OPERA | SAFARI |
---|---|
基本支持 | 49 |
- | 14 |
MOBILE | OPERA MOBILE |
------------------------- | ---------------------- |
基本支持 | 49 |
- | - |
# 解构
解构是 ECMAScript 6 中的新特性,解构能够使我们从数组和对象中提取值并且把这些值赋 到一些变量上,这些变量的组成语法类似于数组和对象。这种语法是清晰明确和易于理解, 并且当传递参数给函数时特别有用。
在 ECMAScript 5 中,配置对象经常被用来处理大量的可选参数,特别是当属性的顺序并不 重要时。思考这个函数:
function initiateTransfer(options) {
var protocol = options.protocol,
port = options.port,
delay = options.delay,
retries = options.retries,
timeout = options.timeout,
log = options.log;
// code to initiate transfer
}
options = {
protocol: 'http',
port: 800,
delay: 150,
retries: 10,
timeout: 500,
log: true,
};
initiateTransfer(options);
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
这种模式 JavaScript 的开发者经常使用,并且效果非常好,但是我们必须得看函数的内部 才知道它需要哪些参数。使用解构参数,在函数声明时,我们可以清晰地表明需要哪些参数 :
function initiateTransfer({ protocol, port, delay, retries, timeout, log }) {
// code to initiate transfer
}
var options = {
protocol: 'http',
port: 800,
delay: 150,
retries: 10,
timeout: 500,
log: true,
};
initiateTransfer(options);
2
3
4
5
6
7
8
9
10
11
12
在这个函数中,我们使用了对象解构模式替代了配置对象。这使得我们的函数不仅更简洁, 而且易读性更强。
我们还可以将解构参数和普通参数结合使用:
function initiateTransfer(
param1,
{ protocol, port, delay, retries, timeout, log }
) {
// code to initiate transfer
}
initiateTransfer('some value', options);
2
3
4
5
6
7
注意,在参数调用时省略参数将会抛出类型错误:
function initiateTransfer({ protocol, port, delay, retries, timeout, log }) {
// code to initiate transfer
}
initiateTransfer(); // TypeError: Cannot match against 'undefined' or 'null'
2
3
4
当我们需要参数需要时,这是预期的行为,但如果我们希望他们是可选的呢?为了防止缺省 参数时出现这个错误,我们需要给解构参数赋一个默认值:
function initiateTransfer({
protocol,
port,
delay,
retries,
timeout,
log,
} = {}) {
// code to initiate transfer
}
initiateTransfer(); // no error
2
3
4
5
6
7
8
9
10
11
在这个函数中,一个空对象作为默认值赋值给了解构参数。现在,如果这个函数不带任何参 数被调用,并不会报错。
我们也可以给每个解构参数赋默认值:
function initiateTransfer({
protocol = 'http',
port = 800,
delay = 150,
retries = 10,
timeout = 500,
log = true,
}) {
// code to initiate transfer
}
2
3
4
5
6
7
8
9
10
在这个示例中,每一个属性都有一个默认值参数,省去了需要我们手动检查未定义的参数和 在函数内部赋默认值。
# 浏览器对默认解构的支持
桌面浏览器 | | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE | OPERA |
SAFARI |
---|
基本支持 |
14 |
OPERA MOBILE |
------------------ |
基本支持 |
浏览器支持详 情点我查看 (opens new window)。
# 参数传递
有两种方法可以将参数传递给函数:通过引用或者值。修改引用参数具有全局的体现,但是 修改值参数仅仅在函数内部有所体现。
在某些编程语言中,比如 Visual Basic 和 Powershell,我们有指定通过引用还是值来传 递参数的选项,但 JavaScript 并没有。
# 值参数
从技术上讲,JavaScript 只能通过值传递。当我们传递一个值参数给函数时,该值会在函 数的作用域内创建一个副本。因此,对值的任何更改只会在函数内部有所体现。思考这个示 例:
var a = 5;
function increment(a) {
a = ++a;
console.log(a);
}
increment(a); // 6
console.log(a); // 5
2
3
4
5
6
7
在这里,修改函数内部的参数对原始值没有影响。因此,在函数外部变量被记录时,打印的
值仍然是5
。
# 引用参数
在 JavaScript 中,一切都是通过值传递的,但是当我们将一个引用对象(包含数组)变量 作为参数传递时,“值”是对象的引用,更改引用对象的属性会更改原来的对象。
思考这个函数:
function foo(param) {
param.bar = 'new value';
}
obj = {
bar: 'value',
};
console.log(obj.bar); // value
foo(obj);
console.log(obj.bar); // new value
2
3
4
5
6
7
8
9
正如你所看到的,函数内部的对象属性被修改,但修改后的值在函数外部是可见的。
当我们传递一个非基本值时,如数组或对象,在程序背后就会创建一个变量并指向内存中的 原始对象的位置。然后这个变量传递给函数,并且修改它就会影响到原始对象。
# 类型检查和缺省或额外的参数
在强类型语言中,我们必须在函数声明中指定参数的类型,但是 JavaScript 缺乏此特性。 在 JavaScript 中,我们传递给函数的参数类型或参数数量都无关紧要。
假设我们有一个只接受一个参数的函数。当我们调用这个函数时,我们无法限制只能传入一 个参数,我们可以传递一个,两个或更多参数!我们甚至可以不传入任何参数,并且函数不 会报错。
实参和形参的数量有两种不同:
实参的数量少于形参
缺少的参数将等于
undefined
。实参的数量大于形参
额外的参数将被忽略,但是可以通过特殊数组比如变量 arguments 来检索(稍后讨论) 。
# 必要参数
如果函数调用时缺省一个参数,那么这个参数将被设置为undefined
。我们可以利用这种
行为,如果省略参数抛出一个错误:
function foo(mandatory, optional) {
if (mandatory === undefined) {
throw new Error('Missing parameter: mandatory');
}
}
2
3
4
5
在 ECMAScript 6 中,我们可以更进一步,使用默认参数来设置必要参数:
function throwError() {
throw new Error('Missing parameter');
}
function foo(param1 = throwError(), param2 = throwError()) {
// do something
}
foo(10, 20); // ok
foo(10); // Error: missing parameter
2
3
4
5
6
7
8
# 参数对象
rest 参数被添加到 ECMAScript 4 用来替换arguments
对象,但是 ECMAScript 4 没有实
现。随着 ECMAScript 6 的发布,JavaScript 现在正式支持 rest 参数,同时还否决了删
除arguments
对象的计划。
arguments
对象是一个类数组对象,可以在所有函数中使用。它允许参数的值通过索引而
不是名称传递给函数。该对象允许我们将任何数量的参数传递给函数。思考下面的代码:
function checkParams(param1) {
console.log(param1); // 2
console.log(arguments[0], arguments[1]); // 2 3
console.log(param1 + arguments[0]); // 2 + 2
}
checkParams(2, 3);
2
3
4
5
6
这个函数预计只接受一个参数。当我们使用两个参数调用它时,第一个参数通过参数名
称param1
或者参数对象arguments[0]
在函数中是可以访问的,但是第二个参数只能通
过arguments[1]
来访问。此外注意,arguments
对象可以和参数名称一起使用。
arguments
对象包含传递给函数的所有参数,并第一个参数从索引 0 开始。如果我们想在
上面的示例中访问更多的参数,我们会写arguments[2]
和arguments[3]
等。
我们甚至可以完全跳过设置命名参数,仅仅使用arguments
对象:
function checkParams() {
console.log(arguments[1], arguments[0], arguments[2]);
}
checkParams(2, 4, 6); // 4 2 6
2
3
4
事实上,命名参数是为了方便,而不是必要的。类似地,rest 参数也可以用来体现所传递 的参数:
function checkParams(...params) {
console.log(params[1], params[0], params[2]); // 4 2 6
console.log(arguments[1], arguments[0], arguments[2]); // 4 2 6
}
checkParams(2, 4, 6);
2
3
4
5
arguments
对象是一个类数组对象,它缺少数组的方法,比如slice()
和foreach()
。
为了在arguments
对象上使用数组的方法,该对象首先需要转换为一个真正的数组:
function sort() {
var a = Array.prototype.slice.call(arguments);
return a.sort();
}
sort(40, 20, 50, 30); // [20, 30, 40, 50]
2
3
4
5
在这个函数中,Array.prototype.slice.call()
作为一个快速方法将arguments
对象转
换为一个数组。接下来,sort()
方法排序数组中的项并返回了它。
ECMAScript 6 有一个更简单的方法。Array.from()
是 ECMAScript 6 中新增的,它可以
把一个数组对象创建成一个数组:
function sort() {
var a = Array.from(arguments);
return a.sort();
}
sort(40, 20, 50, 30); // [20, 30, 40, 50]
2
3
4
5
# length 属性
虽然参数对象不是数组,但是它拥有length
属性,这可以用来检查传入函数中的参数数量
:
function countArguments() {
console.log(arguments.length);
}
countArguments(); // 0
countArguments(10, null, 'string'); // 3
2
3
4
5
通过使用length
属性,我们可以更好的控制传递给函数的参数数量。例如,如果函数需要
两个参数工作,我们可以使用length
属性来检查传入的参数的数量,如果传入的参数数量
比预期要少,则抛出错误:
function foo(param1, param2) {
if (arguments.length < 2) {
throw new Error('This function expects at least two arguments');
} else if (arguments.length === 2) {
// do something
}
}
2
3
4
5
6
7
rest 参数是数组,所以他们有一个length
属性。在 ECMAScript 中 6 中,前面的代码可
以使用 rest 参数这么来改写:
function foo(...params) {
if (params.length < 2) {
throw new Error('This function expects at least two arguments');
} else if (params.length === 2) {
// do something
}
}
2
3
4
5
6
7
# callee 和 caller 属性
callee
属性是指当前正在运行的函数,caller
是指调用了当前函数的执行函数。在
ECMAScript 5 严格模式下,这些属性已被废弃,如果尝试访问这些属性会导致类型错误。
arguments.callee
属性在递归函数(递归函数是指有规律的自己调用自己的函数)中是有
用的,尤其是当函数名称不可用(即匿名函数)时。因为匿名函数没有名称,只能通
过arguments.callee
属性调用自己。
var result = (function (n) {
if (n <= 1) {
return 1;
} else {
return n * arguments.callee(n - 1);
}
})(4); // 24
2
3
4
5
6
7
# 严格模式和非严格模式中的参数对象
在 ECMAScript 5 非严格模式下,arguments
对象有一个不同寻常特性:它可以与相应的
命名参数的值保持同步。
思考下面的代码:
function foo(param) {
console.log(param === arguments[0]); // true
arguments[0] = 500;
console.log(param === arguments[0]); // true
return param;
}
foo(200); // 500
2
3
4
5
6
7
在这个函数中,一个新值赋给了arguments[0]
。因为arguments
的值始终和相应的命名
参数保持同步,改变arguments[0]
的值也将改变param
的值。事实上,他们像两个不同
名称的相同变量。在 ECMAScript 5 严格模式下,这个令人困惑的arguments
对象的行为
已被删除:
'use strict';
function foo(param) {
console.log(param === arguments[0]); // true
arguments[0] = 500;
console.log(param === arguments[0]); // false
return param;
}
foo(200); // 200
2
3
4
5
6
7
8
此时,改变arguments[0]
并不会影响param
,并且输出结果和预期的一样。该函数的输出
结果在 ECMAScript 6 中和 ECMAScript 5 严格模式下是相同的,但是记住,当函数声明中
使用默认值时,arguments
对象并不受影响:
function foo(param1, param2 = 10, param3 = 20) {
console.log(param1 === arguments[0]); // true
console.log(param2 === arguments[1]); // true
console.log(param3 === arguments[2]); // false
console.log(arguments[2]); // undefined
console.log(param3); // 20
}
foo('string1', 'string2');
2
3
4
5
6
7
8
在这个函数中,即使param3
有默认值,它也和arguments[2]
不相等,因为只有两个参数
传递给了函数。换句话说,设置默认值并不影响arguments
对象。
# 总结
ECMAScript 6 给 JavaScript 带来了大大小数以百计的的改进。越来越多的开发者使用 ECMAScript 6 的特性,很快这些特性不可避免会被使用到。在本教程中,我们已经学会了 ECMAScript 6 是如何升级 JavaScript 中的参数应用,这仅仅只是 ECMAScript 6 中的一 小块表面知识。ECMAScript 6 中还有很多有趣的新特性值得你一看。
# 链接
- ECMAScript 6 兼容性表格 (opens new window), Juriy Zaytsev
- “ECMAScript 2015 语言规范 (opens new window),” ECMA International
封面图片来源 :JS Arguments & Parameters (opens new window)
- 本文章翻译 自How To Use Arguments And Parameters In ECMAScript 6 (opens new window)。
- 本人英文水平有限,翻译不正确不通顺的地方,敬请指出。