zoukankan      html  css  js  c++  java
  • JavaScript 新旧替换三:参数转换

    目录

    引子

    在 ES2015 之前,有把函数的 arguments 转变为某种可以当作数组来使用的方法,现在可以摆脱这些方法了。

    这是继 JavaScript 新旧替换二:赋值和取值的第三篇。

    ES5 方式

    主要是使用了 apply() 方法,该方法用途是在特定的作用域中调用函数,实际上等于设置函数体内 this 对象的值。该接受两个参数:一个是在其中运行函数的作用域,另一个是参数数组。其中,第二个参数可以是 Array 的实例,也可以是 arguments 对象。

    function foo() {
      var args = Array.prototype.slice.apply(arguments);
      console.info(args); // [1,2,3,4,5]
      console.info(args instanceof Array); // true
    }
    foo(1,2,3,4,5);
    

    ES2015+ 方式

    在 ES2015 中引入了一个新的运算符 ...,称为 spreadrest(展开或收集)运算符。不同情况下,会有不同的特性。

    function foo(...args) {
      console.info(args); // [1,2,3,4,5]
      console.info(...args); // 1 2 3 4 5
    }
    foo(1,2,3,4,5);
    

    这里运算符 ... 使用的特性有:

    • foo(...args) 中使用了收集的特性,把一系列值收集到一起成为一个数组。
    • console.info(...args) 中使用了展开的特性,这里把数组 args 展开为一组函数调用的参数。

    在 ES2018 中这个运算符引入到了对象。它会取出对象的所有可遍历属性,拷贝到当前对象中。

    使用于对象常见的几种情况如下:

    一般对象

    let obj = {a:1,b:2};
    let objCopy = {...obj};
    objCopy.a = 2;
    console.info(obj); // {a:1,b:2}
    console.info(objCopy); // {a:2,b:2}
    
    let complexObj = {a:{b:1},c:2};
    let complexObjCopy = {...complexObj};
    complexObjCopy.a.b = 2;
    console.info(complexObj); // {a:{b:2},b:2}
    console.info(complexObjCopy); // {a:{b:2},b:2}
    

    可见只是进行了浅拷贝,等同于使用 Object.assign() 方法。

    数组

    let objArray = {...['a','b']};
    console.info(objArray); // {0: "a", 1: "b"}
    
    let complexObjArray = {...['a',['b'],'c']};
    console.info(complexObjArray); // {0: "a", 1: ["b"],2:"c"}
    

    字符串

    let objStr = {...'hi'};
    console.info(objStr); // {0: "h", 1: "i"}
    

    其它

    {...1} // {}
    
    {...true} // {}
    
    {...undefined} // {}
    
    {...null} // {}
    
    {...NaN} // {}
    

    应用

    运算符 ... 较常见应用有:

    复制对象或数组

    let obj = {a:{b:1},c:2};
    let objCopy = {...obj};
    objCopy.a.b = 2;
    console.info(obj); // {a:{b:2},b:2}
    console.info(objCopy); // {a:{b:2},b:2}
    
    let arr = ['1',['2']];
    let arrCopy = [...arr];
    arrCopy[1][0] = '3';
    console.info(arr); // ['1',['3']]
    console.info(arrCopy); // ['1',['3']]
    

    需要注意都是浅拷贝。

    合并对象或数组

    let objA = {a:1}, objB = {b:2};
    let obj = {...objA,...objB};
    console.info(obj); // {a:1,b:2}
    
    let arrA = ['1'], arrB = ['2'];
    let arr = [...arrA,...arrB];
    console.info(arr); // ['1',2']
    

    与解构结合

    与对象解构结合

    与对象解构结合需要满足的条件有:

    • 等号右边是一个对象。
    • 解构必需是最后一个参数,否则会报错。
    let {a,...b} = undefined; // Uncaught TypeError: Cannot destructure property `a` of 'undefined' or 'null'.
    
    let {...q,k} = {m:1,n:2,k:3}; // Uncaught SyntaxError: Rest element must be last element
    
    let {x,...z} = {x:1,y:2,z:3};
    z // {y:2,z:3}
    

    使用了扩展运算符后,把目标对象上所有可遍历但尚未被读取的属性,分配到指定对象上。

    与数组解构结合

    与数组解构结合需要满足的条件有:

    • 等号右边是一个数组。
    • 解构必需是最后一位,否则会报错。
    let [...x] = 123; // 123 is not iterable
    
    let [...q,k] = [1,2,3]; // Uncaught SyntaxError: Rest element must be last element
    
    let [x,...z] = [1,2,3];
    z // [2,3]
    

    参考资料

  • 相关阅读:
    Li Fei-fei写给她学生的一封信,如何做好研究以及写好PAPER(转载)
    OpenCV学习-——OpenCV永久配置(win7 64位+VS2012+OpenCV 2.4.6)
    Eclipse + Jdk配置Java编程环境
    BeautifulSoup安装与引用
    自动启动服务
    自动行列转置
    java-接口
    java基础-抽象类
    JAVA学习day 05 运算
    JAVA学习day 04 变量
  • 原文地址:https://www.cnblogs.com/thyshare/p/12906100.html
Copyright © 2011-2022 走看看