原文地址:JavaScript & The spread operator

拓展运算符看起来像什么? 三个点,...

它能做什么? 拓展运算符允许一个表达式在某个地方展开成为多个元素、变量或参数

定义看上去毫无乐趣,下面通过几个例子理解到底什么是拓展运算符。

初识

  • 1、创建一个数组middle
  • 2、创建第二个数组,包含第一个数组
  • 3、输出结果
var middle = [3, 4];
var arr = [1, 2, middle, 5, 6]; console.log( arr );
// [1, 2, [3, 4], 5, 6]

上面没有使用拓展运算符arr在创建的时候把middle插入到arr数组的中间,此时arr数组里面包了一个数组middle

如果我们的目标是两个数组,那上面的做法很不错,但如果我们只想要一个数组呢?

我们可以通过拓展运算符实现。再提一次,拓展运算符允许数组的元素展开。来看下面的代码,除了使用了拓展运算符,代码和上面的没啥区别:

var middle = [3, 4];
var arr = [1, 2, ...middle, 4, 5]; console.log( arr )
// [1, 2, 3, 4, 5, 6]

还记得刚才读到的拓展运算符的定义吗?上面的代码发挥了拓展运算符的作用,正如你看的,我们创建了数组arr,然后对middle使用了拓展运算符,此时,middle数组并没有直接插入到arr中,而是被展开了。middle数组中的每一个元素插入到了arr数组中。此时,arr不再是嵌套数组,而是一个从1到6的简单数组。

1、Slice

slice是JavaScript中数组方法,允许我们拷贝数组。同样,我们可以使用拓展运算符来拷贝数组。

var arr = ['a', 'b', 'c'];
var arr2 = [...arr]; console.log(arr2);
// ['a', 'b', 'c']

瞧,我们成功复制了数组。arr数组中的所有元素都分解成独立的个体,然后分配到arr2数组中。我们现在可以对arr2进行任意修改,都不会影响到源数组arr

上述可以工作的原因是arr的值填充到了arr2定义的方括号中。所以,我们是让arr2的值和arr的值相等,而不是和arr本身相等。下面的不适用拓展运算符的示例更能说明问题。

如果我们简单的创建一个数组,然后让另一个数组和他相等,就像这样:

var arr = ['a', 'b', 'c'];
var arr2 = arr; console.log(arr2);
// ['a', 'b', 'c']

现在你可能会说,这也work了,和上面的结果一样啊。

然后,这只是看起来一样,但是本质是完全不同。我们并没有让arr2的值和arr中每个值相同,而是直接让两个数组相等,这也就意味着任何对arr2的修改都是影响arr

arr2.push('d');

console.log(arr2);
// ['a', 'b', 'c', 'd'] console.log(arr);
// ['a', 'b', 'c', 'd']

正如你亲眼所见,即使没有明确的向源数组arr添加新值,新值d依然会添加到arr中。

2、Concat

使用concat()可以连接数组,使用拓展运算符也能实现。首先看看使用concat()是什么样子:

var arr = ['a', 'b', 'c'];
var arr2 = ['d', 'e', 'f']; arr1 = arr.concat(arr2);
console.log(arr);
// ['a', 'b', 'c', 'd', 'e', 'f']

现在,使用拓展运算符:

var arr = ['a', 'b', 'c'];
var arr2 = ['d', 'e', 'f']; arr = [...arr, ...arr2];
console.log(arr);
// ['a', 'b', 'c', 'd', 'e', 'f']

结果完全相同,每个数组都分解然后加入到新的数组arr中。

注:原文评论中有人指出使用拓展运算符连接数组可能并不高效。在测试中,使用拓展运算符要比concat慢一些。对大数组的处理,使用拓展运算符可能并不理想。

3、Math

Math函数可以与拓展运算符结合使用,我们以Math.max()为例。

简单介绍一下,Math.max()返回一组数字中的最大值。

Math.max();
// -Infinity Math.max(1, 2, 3);
// 3 Math.max(100, 3, 4);
// 100

如果没有拓展运算符,我们也可以借助.apply()通过Math.max()求数组中的最大值:

var arr = [2, 4, 8, 6, 0];

function max(arr) {
return Math.max.apply(null, arr);
} console.log(max(arr));
// 8

这是让人讨厌的实现。

现在,我们来看通过拓展运算符如何实现。代码更加简洁:

var arr = [2, 4, 8, 6, 0];
var max = Math.max(...arr); console.log(max);
// 8

4、String to Array

将字符串转成数组有很多种方法,比如slicesplit等,使用拓展运算符会更加简单:

var str = "hello";
var chars = [...str]; console.log(chars);
// ['h', 'e',' l',' l', 'o']

很酷吧,拓展运算符会遍历str中的每个字符,然后分配到chars新数组中。

通过四个例子理解JavaScript拓展运算符的更多相关文章

  1. 最简单的例子理解Javascript闭包

    理解Javascript的闭包非常关键,本篇试图用最简单的例子理解此概念. function greet(sth){ return function(name){ console.log(sth + ...

  2. 深入理解javascript作用域系列第四篇——块作用域

    × 目录 [1]let [2]const [3]try 前面的话 尽管函数作用域是最常见的作用域单元,也是现行大多数javascript最普遍的设计方法,但其他类型的作用域单元也是存在的,并且通过使用 ...

  3. 深入理解javascript作用域系列第四篇

    前面的话 尽管函数作用域是最常见的作用域单元,也是现行大多数javascript最普遍的设计方法,但其他类型的作用域单元也是存在的,并且通过使用其他类型的作用域单元甚至可以实现维护起来更加优秀.简洁的 ...

  4. 深入理解javascript原型和闭包(1)——一切都是对象

    “一切都是对象”这句话的重点在于如何去理解“对象”这个概念. ——当然,也不是所有的都是对象,值类型就不是对象. 首先咱们还是先看看javascript中一个常用的函数——typeof().typeo ...

  5. 深入理解JavaScript中创建对象模式的演变(原型)

    深入理解JavaScript中创建对象模式的演变(原型) 创建对象的模式多种多样,但是各种模式又有怎样的利弊呢?有没有一种最为完美的模式呢?下面我将就以下几个方面来分析创建对象的几种模式: Objec ...

  6. 理解Javascript的动态语言特性

    原文:理解Javascript的动态语言特性 理解Javascript的动态语言特性 Javascript是一种解释性语言,而并非编译性,它不能编译成二进制文件. 理解动态执行与闭包的概念 动态执行: ...

  7. JavaScript 扩展运算符

    扩展运算符格式扩展运算符格式很简单,就是三个点(...) 扩展运算符作用???扩展运算符允许一个表达式在期望多个参数(用于函数调用)或多个元素(用于数组字面量)或多个变量(用于解构赋值)的位置扩展. ...

  8. 深入理解JavaScript系列(15):函数(Functions)

    介绍 本章节我们要着重介绍的是一个非常常见的ECMAScript对象——函数(function),我们将详细讲解一下各种类型的函数是如何影响上下文的变量对象以及每个函数的作用域链都包含什么,以及回答诸 ...

  9. 深入理解JavaScript系列(13):This? Yes,this!

    介绍 在这篇文章里,我们将讨论跟执行上下文直接相关的更多细节.讨论的主题就是this关键字.实践证明,这个主题很难,在不同执行上下文中this的确定经常会发生问题. 许多程序员习惯的认为,在程序语言中 ...

随机推荐

  1. HDU 5914 Triangle(打表——斐波那契数的应用)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5914 Problem Description Mr. Frog has n sticks, whos ...

  2. NopCommerce用core重写ef

    最近看了NopCommerce源码,用core学习着写了一个项目,修改的地方记录下.项目地址 NopCommerce框架出来好久了.18年的第一季度 懒加载出来后也会全部移动到.net core.那么 ...

  3. 规范的python编码

    规范的 python 编码令人赏心悦目,令代码的表达逻辑更清晰,使得工程代码更容易被维护和交流: 编码规范包括对于代码书写格式的约束,不良语法的禁用和推荐的编码手法,下面做些简要的描述: 1. 代码规 ...

  4. Vue版本过渡变化

    到了2.0以后,有哪些变化: 在每个组件模板,不在支持片段代码 之前: <template id="aaa"> <h3>我是组件</h3>< ...

  5. destoon各栏目调用汇总

    ================================================================== destoon各栏目调用汇总 ================== ...

  6. 使用layui-tree美化左侧菜单,点击生成tab选项

    layui-tree美化左侧菜单 html <div class="layui-side layui-bg-black"> <div class="la ...

  7. Spring AOP中 args和arg-names的区别

    这两天在看aop aspectj的各种语法,发现里面有两个概念 args和arg-names很容易混淆,网上也基本没说清楚,所以就动手试了一下,发现还是自己试试比较好理解 先说结论: args是和ex ...

  8. Windows核心编程&进程

    1. 进程的定义 说白了进程就是一个正在运行的执行程序,包含内核对象和独立的地址空间,内核对象负责统计和管理进程信息,地址空间包括所有可执行文件或DLL 模块的代码和数据.动态内存分配(线程堆和栈的分 ...

  9. JavaScript 函数创建思想

    //定义一个函数的步骤//1.开辟一个新的空间地址//2.把函数体里面的代码当做字符串存储到空间里面(一个函数如果只定义了,没有执行的话,这个函数没有任何意义)//3.在把我们的地址给我们的函数名fu ...

  10. 【转】GPS静态观测网的设计指标

     GPS网的设计指标是指导GPS网设计量化因子,是评价GPS网设计优劣的数值标准.评价GPS网设计的优劣主要从以下三个因素考虑:1.质量(包括精度和可靠性):2.效率:3.费用. 一.GPS网设计的精 ...