当前位置:首页 > 技术分析 > 正文内容

ES6!你没用过的新语法(附代码)

ruisui883个月前 (03-17)技术分析35

ECMAScript 6(以下简称ES6)是JavaScript语言的下一代标准,已经在2015年6月正式发布了。Mozilla公司将在这个标准的基础上,推出JavaScript 2.0。ECMA Script,JavaScript的语言标准。至今已经发布5年多了,但是因为蕴含的语法之广,完全消化需要一定的时间,这里我总结了部分ES6,以及ES6以后新语法的知识点,使用场景,希望对各位有所帮助

一、let和const

let

用法类似于var,但是所声明的变量,只在let命令所在的代码块内有效,即let声明的是一个块作用域内的变量。

特点:

  • 不存在变量提升。
  • 暂时性死区——只要块级作用域内存在let命令,它所声明的变量就“绑定”(binding)这个区域,不再受外部的影响。
  • 不允许重复声明。
  • 块级作用域——被{}包裹,外部不能访问内部。

应用案例与分析:

// 使用var

for (var i = 0; i < 5; i++) {

setTimeout(function () {

console.log(i);

});

} // => 5 5 5 5 5

// 使用let

for (let i = 0; i < 5; i++) {

setTimeout(function () {

console.log(i);

});

} // => 0 1 2 3 4

上面使用let的代码中,变量i是let声明的,当前的i只在本轮循环有效,所以每一次循环的i其实都是一个新的变量,JavaScript 引擎内部会记住上一轮循环的值,初始化本轮的变量i时,就在上一轮循环的基础上进行计算所以最后能正常输出i的值。

注意:

for循环还有一个特别之处,就是设置循环变量的那部分是一个父作用域,而循环体内部是一个单独的子作用域,所以我们可以在循环体内部访问到i的值。

let和var全局声明时,var可以通过window的属性访问而let不能。

const

const声明一个只读的常量。一旦声明,常量的值就不能改变。const实际上保证的是变量指向的那个内存地址所保存的数据不得改动。对于简单类型的数据(数值、字符串、布尔值),值就保存在变量指向的那个内存地址,因此等同于常量。但对于复合类型的数据(主要是对象和数组),变量指向的内存地址,保存的只是一个指向实际数据的指针,const只能保证这个指针是固定的,至于它指向的数据结构是不是可变的,就完全不能控制了。因此,将一个对象声明为常量必须非常小心。

因此,我们使用const时,不能只声明而不初始化值,否则会报错:

const a;

// SyntaxError: Missing initializer in const declaration

const的其他特性和let很像,一般推荐用它来声明常量,并且常量名大写。

二、数组的扩展

扩展运算符

扩展运算符(spread)是三个点(...),将一个数组转为用逗号分隔的参数序列

应用场景:

1 复制数组

const a1 = [1, 2];

const a2 = [...a1];

2 合并数组

const arr1 = ['1', '2'];

const arr2 = ['c', {a:1} ];

// ES6 的合并数组

[...arr1, ...arr2]

注:这两种方法都是浅拷贝,使用的时候需要注意。

3 将字符串转化为数组

使用扩展运算符能够正确识别四个字节的 Unicode 字符。凡是涉及到操作四个字节的 Unicode 字符的函数,都有这个问题。因此,最好都用扩展运算符改写。

[...'xuxi']

// [ "x", "u", "x", "i" ]

4 实现了 Iterator 接口的对象

let nodeList = document.querySelectorAll('div');

let arr = [...nodeList];

上面代码中,querySelectorAll方法返回的是一个NodeList对象。它不是数组,而是一个类似数组的对象。扩展运算符可以将其转为真正的数组,原因就在于NodeList对象实现了 Iterator 。

Array.from()

Array.from方法用于将类对象转为真正的数组:类似数组的对象和可遍历的对象(包括 ES6 新增的数据结构 Set 和 Map)。

实际应用中我们更多的是将Array.from用于DOM 操作返回的 NodeList 集合,以及函数内部的arguments对象。

// NodeList对象

let nodeList = document.querySelectorAll('p')

let arr = Array.from(nodeList)

// arguments对象

function say() {

let args = Array.from(arguments);

}

Array.from还可以接受第二个参数,作用类似于数组的map方法,用来对每个元素进行处理,将处理后的值放入返回的数组。

Array.from([1, 2, 4], (x) => x + 1)

// [2, 3, 5]

Array.of()

Array.of方法用于将一组值,转换为数组。Array.of基本上可以用来替代Array()或new Array(),并且不存在由于参数不同而导致的重载。它的行为非常统一。

Array.of() // []

Array.of(undefined) // [undefined]

Array.of(2) // [21]

Array.of(21, 2) // [21, 2]

数组实例方法includes()

Array.prototype.includes方法返回一个布尔值,表示某个数组是否包含给定的值。该方法的第二个参数表示搜索的起始位置,默认为0。如果第二个参数为负数,则表示倒数的位置,如果这时它大于数组长度(比如第二个参数为-4,但数组长度为3),则会重置为从0开始。

[1, 4, 3].includes(2) // true

[1, 2, 4].includes(3) // false

[1, 5, NaN, 6].includes(NaN) // true

三、对象的扩展

对象的扩展运算符

对象的扩展运算符(...)用于取出参数对象的所有可遍历属性,拷贝到当前对象之中,等同于使用Object.assign()方法。

let a = {

w: 'xu', y: 'xi'

}l

et b = {

name: '12'}let ab = {

...a, ...b

};

// 等同于

let ab = Object.assign({}, a, b);

Object.is()

用来比较两个值是否严格相等,与严格比较运算符(===)的行为基本一致;不同之处只有两个:一是+0不等于-0,二是NaN等于自身。

+0 === -0 //true

NaN === NaN // false

Object.is(+0, -0) // false

Object.is(NaN, NaN) // true

Object.assign()

用于对象的合并,将源对象的所有可枚举属性,复制到目标对象; 如果只有一个参数,Object.assign会直接返回该参数; 由于undefined和null无法转成对象,所以如果它们作为参数,就会报错; 其他类型的值(即数值、字符串和布尔值)不在首参数,也不会报错。但是,除了字符串会以数组形式,拷贝入目标对象,其他值都不会产生效果。

// 合并对象

const target = { a: 1, b: 1 };

const source1 = { b: 2, c: 2 };

const source2 = { c: 3 };

Object.assign(target, source1, source2);

target // {a:1, b:2, c:3}

// 非对象和字符串的类型将忽略

const a1 = '123';

const a2 = true;

const a3 = 10;

const obj = Object.assign({}, a1, a2, a3);

console.log(obj); // { "0": "1", "1": "2", "2": "3" }

注意点:

Object.assign方法实行的是浅拷贝,而不是深拷贝。也就是说,如果源对象某个属性的值是对象,那么目标对象拷贝得到的是这个对象的引用。

对于嵌套的对象,遇到同名属性,Object.assign的处理方法是替换,而不是添加

Object.assign可以用来处理数组,但是会把数组视为对象。

Object.assign([1, 2, 3], [4, 5])

// [4, 5, 3]

Object.assign只能进行值的复制,如果要复制的值是一个取值函数,那么将求值后再复制。

const a = {

get num() { return 1 }

};

const target = {};

Object.assign(target, a)

// { num: 1 }

应用场景:

  • 为对象添加属性和方法
  • 克隆/合并对象
  • 为属性指定默认值

Object.keys()

返回一个数组,成员是参数对象自身的(不含继承的)所有可遍历属性的键名。

const obj = { 100: '1', 2: '2', 7: '3' };

Object.values(obj)

// ["100", "2", "7"]

Object.values()

返回一个数组,成员是参数对象自身的(不含继承的)所有可遍历属性的键值。注意:返回数组的成员顺序:如果属性名为数值的属性,是按照数值大小,从小到大遍历的。

const obj = { 100: '1', 2: '2', 7: '3' };

Object.values(obj)

// ["2", "3", "1"]

四、set和map数据结构

set

ES6提供了新的数据结构Set,类似于数组,但是成员的值都是唯一的,没有重复的值。Set本身是一个构造函数,用来生成Set数据结构。

实例属性和方法:

  • add(value):添加某个值,返回Set结构本身。
  • delete(value):删除某个值,返回一个布尔值,表示删除是否成功。
  • has(value):返回一个布尔值,表示该值是否为Set的成员。
  • clear():清除所有成员,没有返回值。

s.add(1).add(3).add(3);

// 注意3被加入了两次

s.size // 2

s.has(1) // true

s.has(2) // false

s.delete(3);

s.has(3) // false

可进行遍历操作:

  • keys():返回键名的遍历器
  • values():返回键值的遍历器
  • entries():返回键值对的遍历器
  • forEach():使用回调函数遍历每个成员

Set的遍历顺序就是插入顺序,这个特性有时非常有用,比如使用Set保存一个回调函数列表,调用时就能保证按照添加顺序调用。

应用场景:

// 数组去重

let arr = [1,2,2,3];

let unique = [...new Set(arr)];

// or

function dedupe(array) {

return Array.from(new Set(array));

}

let a = new Set([1, 2, 3]);

let b = new Set([4, 3, 2]);

// 并集

let union = new Set([...a, ...b]);

// Set {1, 2, 3, 4}

// 交集

let intersect = new Set([...a].filter(x => b.has(x)));

// set {2, 3}

// 差集

let difference = new Set([...a].filter(x => !b.has(x)));

// Set {1}

map

类似于对象,也是键值对的集合,各种类型的值(包括对象)都可以当作键。Map结构提供了“值与值”的对应,是一种更完善的Hash结构实现。

实例属性和方法:

  • size属性: 返回Map结构的成员总数。
  • set(key, value): set方法设置key所对应的键值,然后返回整个Map结构。如果key已经有值,则键值会被更新,否则就新生成该键,set方法返回的是Map本身,因此可以采用链式写法。
  • get(key) : get方法读取key对应的键值,如果找不到key,返回undefined。
  • has(key) : has方法返回一个布尔值,表示某个键是否在Map数据结构中。
  • delete(key) : delete方法删除某个键,返回true。如果删除失败,返回false。
  • clear() : clear方法清除所有成员,没有返回值。

遍历方法和set类似,Map结构转为数组结构,比较快速的方法是结合使用扩展运算符(...):

let map = new Map([

[1, 'one'],

[2, 'two'],

[3, 'three'],

]);

[...map.keys()]

// [1, 2, 3]

[...map.values()]

// ['one', 'two', 'three']

[...map.entries()]

// [[1,'one'], [2, 'two'], [3, 'three']]

[...map]

// [[1,'one'], [2, 'two'], [3, 'three']]

数组转map:

new Map([[true, 7], [{foo: 3}, ['abc']]])// Map {true => 7, Object {foo: 3} => ['abc']}

Map转为对象:

function strMapToObj(strMap) {

let obj = Object.create(null);

for (let [k,v] of strMap) {

obj[k] = v;

}

return obj;

}

let myMap = new Map().set('yes', true).set('no', false);

strMapToObj(myMap)

// { yes: true, no: false }

对象转为Map:

function objToStrMap(obj) {

let strMap = new Map();

for (let k of Object.keys(obj)) {

strMap.set(k, obj[k]);

}

return strMap;

}

objToStrMap({yes: true, no: false})

// [ [ 'yes', true ], [ 'no', false ] ]

总结

总的来说,虽然支持es6的情况到目前还不是很乐观,但es6的新语法特性让前端和后端的差异越来越小了,这是一个新时代的开始,我们必须要了解这些新的前沿知识,才能跟上时代的步伐。

扫描二维码推送至手机访问。

版权声明:本文由ruisui88发布,如需转载请注明出处。

本文链接:http://www.ruisui88.com/post/2811.html

分享给朋友:

“ES6!你没用过的新语法(附代码)” 的相关文章

深入理解Vue.js组件通信:父子组件与子父组件数据交互详解

什么是Vue组件通讯 Vue.js 组件通信是指在 Vue 应用的不同组件之间进行数据交换和状态同步的过程。由于 Vue 的组件是基于单文件组件(SFCs)的模块化设计,每个组件都有自己的作用域,因此它们不能直接访问彼此的数据。为了使组件之间能够协同工作,Vue 提供了几种不同的通信方式。以下是 V...

K8s里我的容器到底用了多少内存?

作者:frostchen导语 Linux下开发者习惯在物理机或者虚拟机环境下使用top和free等命令查看机器和进程的内存使用量,近年来越来越多的应用服务完成了微服务容器化改造,过去查看、监控和定位内存使用量的方法似乎时常不太奏效。如果你的应用程序刚刚迁移到K8s中,经常被诸如以下问题所困扰:容器的...

7 招教你轻松搭建以图搜图系统

作者 | 小龙责编 | 胡巍巍当您听到“以图搜图”时,是否首先想到了百度、Google 等搜索引擎的以图搜图功能呢?事实上,您完全可以搭建一个属于自己的以图搜图系统:自己建立图片库;自己选择一张图片到库中进行搜索,并得到与其相似的若干图片。Milvus 作为一款针对海量特征向量的相似性检索引擎,旨在...

three.js cannon.js物理引擎之齿轮动画

今天继续说一说cannon.js物理引擎,并用之前已经学习过的知识实现一个小动画,知识点包括ConvexPolyhedron多边形、Shape几何体、Body刚体、HingeConstraint铰链约束等等知识。因为我之前用纯three.js 的THREEBSP实现过一个静态的齿轮,现在就想配合ca...

详解编程中的同步和异步

本文主要总结一些自己对异步的理解,话不多说 下面开始。一. 单线程 我们常说“JavaScript是单线程的”,所谓单线程,是指在JS引擎中负责解释和执行JavaScript代码的线程只有一个。不妨叫它主线程 但是实际上还存在其他的线程。例如:处理AJAX请求的线程、处理DOM事件的线程、定时器线程...

一篇文章搞懂同步与异步、阻塞与非阻塞

要想掌握好Java NIO需要涉及了解同步与异步、阻塞与非阻塞,本文通过相关例子让你深入理解其本质@mikechen阻塞阻塞与非阻塞是对同一个线程来说的,在某个时刻,线程要么处于阻塞,要么处于非阻塞。阻塞调用是指调用结果返回之前,当前线程会被挂起,调用线程只有在得到结果之后才会返回。举一个例子:当一...