【译】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);
1
2
3
4

在这个函数中,param1param2是函数的形参,传入函数的值(1020)是实参。

# 扩展运算符(...)

在 ECMAScript 5 中,apply()方法传递一个数组作为函数的参数是方便的。例如,常 用Math.max()方法去查找一个数组中的最大值,思考下面这段代码:

var myArray = [5, 10, 50];
Math.max(myArray); // Error: NaN
Math.max.apply(Math, myArray); //50
1
2
3

Math.max()方法不支持数组参数,它仅仅接受数值类型的参数列表。当一个数组传入 到Math.max()函数中时,它抛出一个错误。但是当使用apply()方法后,数组会分解为 单独的数字传入到Math.max()方法中,所以Math.max()方法可以处理它。

幸运的是,随着在 ECMAScript 6 扩展操作符的引入,我们不再需要使用apply()方法。 使用扩展操作符,我们可以轻易地将一个表达式扩展为多个参数:

var myArray = [5, 10, 50];
Math.max(...myArray); // 50
1
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
1
2
3
4
5
6
7

扩展操作符的另一个优点是,它可以很容易地与构造函数一起使用:

new Date(...[2016, 5, 6]); // Mon Jun 06 2016 00:00:00 GMT-0700 (Pacific Daylight Time)
1

当然,我们可以在 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)
1
2

# 浏览器对扩展操作符的支持

桌面浏览器 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE | OPERA |

SAFARI
46
MOBILE
----------------------
46

点我查看 (opens new window)

# Rest 参数

rest 参数的语法与扩展运算符一样,但是它不是展开一个数组到参数中,相反它是将参数 收集为一个数组:

function myFunction(...options) {
  return options;
}
myFunction('a', 'b', 'c'); //["a", "b", "c"]
1
2
3
4

如果没有实参,rest 参数将被设置为一个空数组:

function myFunction(...options) {
  return options;
}
myFunction(); // []
1
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
1
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
1
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
1
2
3
4

另一个限制是,在函数声明中只允许存在一个 rest 参数:

function logArguments(...param1, ...param2) {
}
logArguments(5, 10, 15);    // SyntaxError: parameter after rest parameter
1
2
3

# 浏览器对 rest 参数的支持

桌面浏览器 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE | OPERA |

SAFARI
47
MOBILE
----------------------
47

点我查看 (opens new window)

# 默认参数

# 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
1
2
3
4
5
6
7
8

这个函数预计需要有两个参数,但是当它不带参数调用时,它将使用默认值。在函数内部, 缺少的参数会自动设置为 undefined。因此,我们可以检测到这些参数,并为它们声明为默 认值。检测缺失的参数和设置默认值,我们使用逻辑or操作符(||)。这个操作符检查 表达式的第一个值:如果它 是真值(truthy) (opens new window),操 作符返回它。如果它是假值,操作符返回第二个值。

我们经常在函数中使用这种方式来设置参数的默认值,但是它是有缺陷的。向函数传 入0null也会触发默认值,因为这些参数的值会被转换 为假值(falsy) (opens new window)。因此 ,如果我们真的需要给函数传入0null,我们需要另一种方法来检查是否缺少参数:

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
1
2
3
4
5
6
7
8
9
10
11

在这个函数中,对传入的参数进行类型检查,以确保在赋默认值之前,它们是 undefined。 这种方式需要多写一点的代码,但它是一个更安全的选择,并且允许我们向函数传入参 数0null

# ECMAScript 6 中的默认参数

使用 ECMAScript 6,我们不再需要检查 undefined 值来模拟默认参数了,现在我们可以将 默认值放在函数声明中:

function foo(a = 10, b = 10) {
  console.log(a, b);
}
foo(5); // 5 10
foo(0, null); // 0 null
1
2
3
4
5

正如你所看到的,调用函数时,省略参数会触发默认值,但是传入0null并不会触发 默认值。我们甚至可以使用函数返回值作为参数的默认值:

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)
1
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
1
2
3
4
5
6

你甚至可以在函数声明中执行操作:

function myFunction(a, b = ++a, c = a * b) {
  console.log(c);
}
myFunction(5); // 36
1
2
3
4

注意,与其它语言不同,JavaScript 是在调用函数时计算默认值。

function add(value, array = []) {
  array.push(value);
  return array;
}
add(5); // [5]
add(6); // [6], not [5, 6]
1
2
3
4
5
6

# 浏览器对默认参数的支持

桌面浏览器 | 特性 | CHROME | FIREFOX | INTERNET EXPLORER | MICROSOFT EDGE |

OPERA SAFARI
基本支持 49
- 14
MOBILE OPERA MOBILE
------------------------- ----------------------
基本支持 49
- -

点我查看 (opens new window)

# 解构

解构是 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);
1
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);
1
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);
1
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'
1
2
3
4

当我们需要参数需要时,这是预期的行为,但如果我们希望他们是可选的呢?为了防止缺省 参数时出现这个错误,我们需要给解构参数赋一个默认值:

function initiateTransfer({
  protocol,
  port,
  delay,
  retries,
  timeout,
  log,
} = {}) {
  // code to initiate transfer
}
initiateTransfer(); // no error
1
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
}
1
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
1
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
1
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');
  }
}
1
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
1
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);
1
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
1
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);
1
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]
1
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]
1
2
3
4
5

# length 属性

虽然参数对象不是数组,但是它拥有length属性,这可以用来检查传入函数中的参数数量 :

function countArguments() {
  console.log(arguments.length);
}
countArguments(); // 0
countArguments(10, null, 'string'); // 3
1
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
  }
}
1
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
  }
}
1
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
1
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
1
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
1
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');
1
2
3
4
5
6
7
8

在这个函数中,即使param3有默认值,它也和arguments[2]不相等,因为只有两个参数 传递给了函数。换句话说,设置默认值并不影响arguments对象。

# 总结

ECMAScript 6 给 JavaScript 带来了大大小数以百计的的改进。越来越多的开发者使用 ECMAScript 6 的特性,很快这些特性不可避免会被使用到。在本教程中,我们已经学会了 ECMAScript 6 是如何升级 JavaScript 中的参数应用,这仅仅只是 ECMAScript 6 中的一 小块表面知识。ECMAScript 6 中还有很多有趣的新特性值得你一看。

# 链接

封面图片来源 :JS Arguments & Parameters (opens new window)

Mac电脑Apache虚拟机配置

Mac电脑Apache虚拟机配置

【译】jQuery3新特性

【译】jQuery3新特性