zoukankan      html  css  js  c++  java
  • 深入浅出ES6:不定参数和默认参数

    不定参数

    我们通常使用可变参函数来构造API,可变参函数可接受任意数量的参数。例如,String.prototype.concat方法就可以接受任意数量的字符串参数。ES6提供了一种编写可变参函数的新方式——不定参数。

    我们通过一个简单的可变参数函数containsAll给大家演示不定参数的用法。函数containsAll可以检查一个字符串中是否包含若干个子串,例如:containsAll("banana", "b", "nan")返回true,containsAll("banana", "c", "nan")返回false。

    首先使用传统方法来实现这个函数:

    1 function containsAll(haystack) {
    2   for (var i = 1; i < arguments.length; i++) {
    3     var needle = arguments[i];
    4     if (haystack.indexOf(needle) === -1) {
    5       return false;
    6     }
    7   }
    8   return true;
    9 }

    在这个实现中,我们用到了神奇的arguments对象,它是一个类数组对象,其中包含了传递给函数的所有参数。这段代码实现了我们的需求,但它的可读性却不是最理想的。函数的参数列表中只有一个参数haystack,我们无法一眼就看出这个函数实际上接受了多个参数。另外,我们一定要注意,应该从1开始迭代,而不是从0开始,因为arguments[0]相当于参数haystack。如果我们想要在haystack前后添加另一个参数,我们一定要记得更新循环体。不定参数恰好可以解决可读性与参数索引的问题。下面是用ES6不定参数特性实现的containsAll函数:

    1 function containsAll(haystack, ...needles) {
    2   for (var needle of needles) {
    3     if (haystack.indexOf(needle) === -1) {
    4       return false;
    5     }
    6   }
    7   return true;
    8 }

    这一版containsAll函数与前者有相同的行为,但这一版中使用了一个特殊的...needles语法。我们来看一下调用containsAll("banana", "b", "nan")之后的函数调用过程,与之前一样,传递进来的第一个参数"banana"赋值给参数haystack,needles前的省略号表明它是一个不定参数,所有传递进来的其它参数都被放到一个数组中,赋值给变量needles。对于我们的调用示例而言,needles被赋值为["b", "nan"],后续的函数执行过程一如往常。(注意啦,我们已经使用过ES6中for-of循环。)

    在所有函数参数中,只有最后一个才可以被标记为不定参数。函数被调用时,不定参数前的所有参数都正常填充,任何“额外的”参数都被放进一个数组中并赋值给不定参数。如果没有额外的参数,不定参数就是一个空数组,它永远不会是undefined。

    默认参数

    通常来说,函数调用者不需要传递所有可能存在的参数,没有被传递的参数可由感知到的默认参数进行填充。JavaScript有严格的默认参数格式,未被传值的参数默认为undefined。ES6引入了一种新方式,可以指定任意参数的默认值。

    下面是一个简单的示例 :

    1 function animalSentence(animals2="tigers", animals3="bears") {
    2     return `Lions and ${animals2} and ${animals3}! Oh my!`;
    3 }

    默认参数的定义形式为[param1[ = defaultValue1 ][, ..., paramN[ = defaultValueN ]]],对于每个参数而言,定义默认值时=后的部分是一个表达式,如果调用者没有传递相应参数,将使用该表达式的值作为参数默认值。相关示例如下:

    1 animalSentence();                       // Lions and tigers and bears! Oh my!
    2 animalSentence("elephants");            // Lions and elephants and bears! Oh my!
    3 animalSentence("elephants", "whales");  // Lions and elephants and whales! Oh my!

    默认参数有几个微妙的细节需要注意:

    • 默认值表达式在函数调用时自左向右求值,这一点与Python不同。这也意味着,默认表达式可以使用该参数之前已经填充好的其它参数值。举个例子,我们优化一下刚刚那个动物语句函数:

      1 function animalSentenceFancy(animals2="tigers",
      2     animals3=(animals2 == "bears") ? "sealions" : "bears")
      3 {
      4   return `Lions and ${animals2} and ${animals3}! Oh my!`;
      5 }

      现在,animalSentenceFancy("bears")将返回“Lions and bears and sealions. Oh my!”。

    • 传递undefined值等效于不传值,所以animalSentence(undefined, "unicorns")将返回“Lions and tigers and unicorns! Oh my!”。
    • 没有默认值的参数隐式默认为undefined,所以
    1 function myFunc(a=42, b) {...}

      是合法的,并且等效于

    1 function myFunc(a=42, b=undefined) {...}

    停止使用arguments

    现在我们已经看到了arguments对象可被不定参数和默认参数完美代替,移除arguments后通常会使代码更易于阅读。除了破坏可读性外,众所周知,针对arguments对象对JavaScript虚拟机进行的优化会导致一些让你头疼不已的问题。

    我们期待着不定参数和默认参数可以完全取代arguments,要实现这个目标,标准中增加了相应的限制:在使用不定参数或默认参数的函数中禁止使用arguments对象。曾经实现过arguments的引擎不会立即移除对它的支持,当然,现在更推荐使用不定参数和默认参数。

  • 相关阅读:
    CSP2019 Day2T3 树的重心
    51nod1835 完全图
    [HNOI2015]亚瑟王
    [六省联考2017]分手是祝愿
    洛谷P4550 收集邮票
    06_手工负载分担(数通华为)
    05_配置交换机SSH服务(数通华为)
    04_配置交换机Telnet服务(数通华为)
    03_vlan & access & trunk 口(数通华为)
    02_搭建HTTP客户端与服务器(数通华为)
  • 原文地址:https://www.cnblogs.com/qcxc/p/7238626.html
Copyright © 2011-2022 走看看