这次我们来谈谈数组、函数、对象,这三个家伙出现的频率太高了。在整个js的体系中占据着非常重要的位置。es6对其的优化扩展也是整个es6比较重要的部分。

数组

首先来谈谈数组,划重点开始喽,个人觉得可能简洁写法、解构赋值、扩展运算符,可能出现的频率比较高一点。
    先介绍几个很有用的方法:

Array.from() 转换为数组

将类似数组的对象和可遍历的即有iterator接口的对象转化为真正的数组,在实际应用中dom的nodelist对象和函数中arguments对象是常用的对象。
    另外稍微提及一下,扩展运算符(...)同样可以将某些数据结构转化为数组,但相比Array.from,它只能转换含有iterator接口的结构,而Array.from可以转换所有带有length属性的对象,包括字符串(可以正确处理大于\uFFFF的Unicode字符)。
Array.from({length:3});
//[ undefined, undefined, undefined]
同时,Array.from可以接受第二个参数,作用类似于数组的map方法,用来对每个元素进行处理并将处理后的值放入数组,如果在第二个值中用到了this,还可以接受第三个参数用来绑定this的指向。
    配合三个参数,Array.from实际上实现了把一个原始的数据结构通过一系列函数处理变为数组,进而可以用强大的数组方法来进行数据的处理。

Array.of() 数值转化为数组

Array.of与Array.from有异曲同工之妙,用于将一组值转换为数组,而它又与Array()、New Array()有什么区别呢?
    其实Array.of基本可以替代Array()、New Array(),并且不存在采纳数不同而导致的重载,他的行为是统一的,只会把传入的参数认为是数组成员而不是数组长度。
Array.of(3);//[3]
Array(3);//[,,,]

copyWithin() 数组元素复制

在当前数组内将指定位置的成员复制到其他位置(会覆盖原有成员),返回值是当前数组(修改后的数组),使用这个函数会修改当前数组。

find() findIndex() 寻找数组内元素

find方法用于找出第一个符合条件的数组成员,参数为回调参数,对所有数组成员执行回调函数,直到出现第一个返回值为true的成员,然后返回该成员,如果没有符合条件的成员,就返回undefined。
    findIndex与find十分类似,只是返回值是数组成员位置,如果么有符合成员则返回-1.
    另外两个函数都可以接受第二个参数,用来绑定回调函数的this,而且两个方法都可以用于发现NaN,弥补了indexOf的不足。

fill() 填充数组元素

fiil用于使用给定值填充数组,第一个参数为填充的给定值,第二个第三个参数指定填充的起始位置和结束位置。

entries() keys() values() 遍历数组的三种方法

分别是对键值对、键名、键值的遍历,可用for...of循环。

includes() 数组是否包含给定值

该方法属于es7,第二个参数表示搜索的起始位置

数组空位让数组从此undefined

数组的空位指数组的某一个位置没有任何值,没有任何值不代表是undefined,undefined是有值的。es5中空位并不明确,但es6中明确规定空位转为undefined。但是es5中的各种方法还是会对空位处理不统一,所以建议避免空位。

数组推导强大的简洁写法

es6里在[]里就可以随意写入变量作为成员,es7里数组推导更提供了简洁写法,允许直接通过现有数组等有iterator接口的数据结构生成新数组。
var a1=[1,2,3,4];
var a2=[for(i of a1) i*2];
注意数组推导中,for...of结构总是写在最前面,返回的表达式写在最后面。
    表达式部分比较灵活,可以连用多个if语句筛选,它实际上就可以替代了之前使用的map和filter方法。
    另外需要注意的是,方括号里在数组推导过程中实际上是进行了一个类似函数的行为,所以方括号内是一个单独的作用域;数组推导中,新数组是立即在内存生成,如果原数组过大,将非常消耗内存;字符串可以进行for...of循环,所以和Array.from一样字符串也可以用于数组推导。

函数

es6为函数增加了许多便利的属性和使用的福利,减少了一些不必要的麻烦,也加了许多提升性能的红利。
    下面开始划重点:

函数参数默认值

这方面在本系列博客的前一篇中也有所提及,所以本篇只是粗略讲讲。
    在函数机制里,参数是默认声明的,即可以认为是在函数内部首行隐藏了一条参数定义语句,只是默认值是在函数执行时才会运行,如果给予了正确的参数值,也就是默认值未生效的话,默认值是不会执行的。函数参数默认值可以直接在函数定义时加到参数声明括号里,形如
function(a,b=0){}

function a(a=0,b){}
a(,1)//报错
a(undefined,1) // [1,1]
另外还需要注意几点:
    1.如果在非尾部函数参数定义了默认值,其实是不可以省略的,除非显式的输入undefined,见上面代码。

    2.理论上来说函数的length属性指函数的形参个数,但不严格的说,在设置默认参数后则返回没有指定默认值的形参个数,总体来说,应用函数默认值后,length可能会和我们预期值不相同,length可以理解为期望传入的参数个数,设置了默认值自然就不是被期望的了,但是在chorme的56版测试的length结果是:
    第一个设置默认参数的形参前的形参个数
(function a(a=1,b){}).length    //0
(function a(a,b=1){}).length    //1
(function a(a,c,b=1){}).length  //2
(function a(a,c,d,b=1){}).length    //3
(function a(a,c=2,d,b=1){}).length  //1
3.如果参数默认值是一个变量,那么该变量所处的作用域就是本函数的作用域。

    4.此外默认参数的应用是非常灵活的,比如将默认参数设置为一个执行的函数,如mustBeProvided = throwError(),若不提供则执行函数并返回结果,实际上定义了一个必须传入的参数,否则则报错。

rest参数

rest参数用于获取多余的参数,形如function(...value),实际上可以用value替代arguments,但要注意的是rest参数后不能有其他参数,否则会报错,另外rest参数无法反映到函数的length属性中。

扩展运算符

扩展运算符是是三个点(...),类似于rest参数的逆运算,它是将数组解析成用逗号分隔的参数序列,同样在将扩展运算符运用到赋值中时,只能将其放在最后一位,否则会报错,扩展运算符的应用十分广泛,给参数赋值、数组合并、函数数组返回值、解析数组带来了极大的便利。
let values = [1,2,3,4]
[0,...values]  //[0,1,2,3,4]

箭头函数

十分洋气炫酷的箭头函数(=>),x => x,一行简洁的代码就可以写出原来需要三行的函数functoin (x){return x},如果不需要参数或者多个参数,就用圆括号代表或包裹参数部分。
    另外还有几个注意点:
    1.如果箭头函数的代码块部分多于一条语句,就要使用大括号将其括起来,并使用return语句返回,但是当箭头函数直接返回一个对象时,就需要在对象{}外面加上()。
    2.this的指向。函数体内的this指向定义时所在的对象,而不是使用时所在的对象,其实箭头函数自身没有自己的this,所以内部的this就是外层作用域的this。
    3.不可以当作构造函数,不可以使用new命令
    4.不可以使用arguments对象
    5.不可以使用yield命令
    6.没有super、new.target变量

函数绑定

es7的一个提案,绑定运算符是并排的双冒号(::),双冒号左边是对象,this将绑定在它身上,右边是要执行绑定的函数,返回的结果仍然是原函数,如果左边的对象为空,右边的是一个对象的方法,则this绑定在该对象上。

尾调用优化

尾调用时函数式编程的一个重要概念,指某个函数的最后一步是调用另一个函数,最后一步必须只是单纯地调用函数,下面给出正确写法和一些错误写法
function f(x){
    return g(x);
}

//以上是正确的尾调用写法
//以下是错误的尾调用写法

function f(x){
    y=g(x);
    return y;
}

function f(x){
    g(x);
}

function f(x){
    return g(x)+1;
}
尾调用在递归调用优化里面十分有用,如果在递归调用里面不采用尾调用,那么在每次递归时上一个函数并没有完全执行完毕,还保留着引用在调用栈里面,一旦递归次数过多,那么将会占用很大的内存,而尾递归优化后,每次递归时上一个函数调用下一个函数就是最后一个步骤了,本函数已经结束,所以直接在调用栈里用新函数覆盖旧函数的调用即可。

对象

属性简洁表示方法

和数组一样,es6允许直接写入变量和函数作为对象的属性和方法,下面分别给出属性和方法简写的例子
var foo="bar";
var baz={foo};
baz//{foo:"bar"}

function f(){
    return {x,y};
}

var o={
    method(){//不用累赘的function
        return "hello";
    }
};
var t={o.method};

属性名表达式

可以直接使用表达式标识作为属性名和方法名,表达式放在中括号里面
let a="foo";
let obj={
    [a]:true,
    ["a"+"b"]:false,
    ["h"+"i"](){
        return "hello";
    }
};
但是千万要注意,属性名表达式和简洁表示方法不能同时使用,[]里面不能存放变量,否则会报错。另外如果觉得这样的对象还不够给力,可以看看我后面的博客里介绍的数据结构map,将键名极大扩展。

Object.is() Object.assign()

object.of()用来比较两个值是否严格相等,它与严格比较运算符(===)行为基本一致,不同之处有两个+0和-0不相等,而且NaN与NaN相等。
    Object.assign()用来将源对象的所有可枚举的属性复制到目标对象中,第一个参数为目标对象,后面的都是参数也就是被复制对象。不可枚举的属性和继承的属性不会被复制,复制时后面的属性会覆盖前面的属性。

属性遍历

1.for in遍历,不过会遍历原型链上继承的其他方法,而且顺序可能会有一些问题
2.Object.keys(obj),会返回键名数组
3.Object.getOwnPropertyNames(obj),返回数组,包含自身所有属性,不包含symbol属性,但包含不可枚举属性
4.Object.getOwnPropertySymbols(obj),返回数组,包含对象自身所有symbol属性
5.Reflect.ownKeys(obj),返回一个数组,包含所有属性
6.Reflect.enumerate(obj),返回一个iterator对象,遍历对象自身的和继承的所有可枚举属性,不包含symbol属性,基本与for in类似

最后提一点,es7提案将rest参数和扩展运算符(…)引入对象。