为了保证可读性,本文采用意译而非直译。前端
ECMAScript 2015,也称为ES6,是一个花了6年时间完成的主要版本。从那时起,负责ECMAScript标准开发的技术委员会39 (TC39)每一年都会发布该标准的新版本。这个年度发布周期简化了这个过程,并使新特性快速可用,JavaScript社区对此表示欢迎。git
今年,ECMAScript 2019(简称ES2019)将会发布。 新功能包括Object.fromEntries()
,trimStart()
,trimEnd()
,flat()
,flatMap()
,symbol对象的description属性,可选的catch
绑定等。github
好消息是这些功能已经在最新版本的Firefox和Chrome中实现,而且它们也能够被转换,以便旧版浏览器可以处理它们。正则表达式
在JavaScript中,将数据从一种格式转换为另外一种格式很是常见。 为了便于将对象转换为数组,ES2017引入了Object.entrie()
方法。 此方法将对象做为参数,并以[key,value]
的形式返回对象本身的可枚举字符串键控属性对的数组。 例如:编程
const obj = {one: 1, two: 2, three: 3}; console.log(Object.entries(obj)); // => [["one", 1], ["two", 2], ["three", 3]]
可是若是咱们想要作相反的事情并将键值对列表转换为对象呢? 某些编程语言(如Python)为此提供了dict()
函数。 在Underscore.js
和Lodash中还有_.fromPairs
函数。数组
ES2019引入Object.fromEntries()
方法为JavaScript带来相似的功能, 此静态方法容许你轻松地将键值对列表转换为对象:浏览器
const myArray = [['one', 1], ['two', 2], ['three', 3]]; const obj = Object.fromEntries(myArray); console.log(obj); // => {one: 1, two: 2, three: 3}
如你所见,Object.fromEntries()
与Object.entries()
所作的事情正好相反。 虽然之前能够实现Object.fromEntries()相同的功能,但它实现方式有些复杂:app
const myArray = [['one', 1], ['two', 2], ['three', 3]]; const Array.from(myArray).reduce((acc, [key, val]) => Object.assign(acc, {[key]: val}), {}) console.log(obj); // => {one: 1, two: 2, three: 3}
请记住,传递给Object.fromEntries()
的参数能够是实现可迭代协议的任何对象,只要它返回一个两元素,相似于数组的对象便可。编程语言
例如,在如下代码中,Object.fromEntries()
将Map对象做为参数,并建立一个新对象,其键和对应值由Map中的对给出:函数
const map = new Map(); map.set('one', 1); map.set('two', 2); const obj = Object.fromEntries(map); console.log(obj); // => {one: 1, two: 2}
Object.fromEntries()
方法对于转换对象也很是有用,思考如下代码:
const obj = {a: 4, b: 9, c: 16}; // 将对象转换为数组 const arr = Object.entries(obj); // 计算数字的平方根 const map = arr.map(([key, val]) => [key, Math.sqrt(val)]); // 将数组转换回对象 const obj2 = Object.fromEntries(map); console.log(obj2); // => {a: 2, b: 3, c: 4}
上述代码将对象中的值转换为其平方根。 为此,它首先将对象转换为数组,而后使用map()
方法获取数组中值的平方根,结果是能够转换回对象的数组。
使用Object.fromEntries()
的另外一种状况是处理URL的查询字符串,如本例所示
const paramsString = 'param1=foo¶m2=baz'; const searchParams = new URLSearchParams(paramsString); Object.fromEntries(searchParams); // => {param1: "foo", param2: "baz"}
此代码中,查询字符串将传递给 URLSearchParams()
构造函数。 而后将返回值(即URLSearchParams
对象实例)传递给Object.fromEntries()
方法,结果是一个包含每一个参数做为属性的对象。
Object.fromEntries()
方法目前是第4阶段提案,这意味着它已准备好包含在ES2019标准中。
trimStart()
和trimEnd()
方法在实现与trimLeft()
和trimRight()
相同。这些方法目前处于第4阶段,将被添加到规范中,以便与padStart()
和padEnd()
保持一致,来看一些例子:
const str = " string "; // es2019 console.log(str.trimStart()); // => "string " console.log(str.trimEnd()); // => " string" // 相同结果 console.log(str.trimLeft()); // => "string " console.log(str.trimRight()); // => " string"
对于Web兼容性,trimLeft()
和trimRight()
将保留为trimStart()
和trimEnd()
的别名。
flat()
方法能够将多维数组展平成一维数组
const arr = ['a', 'b', ['c', 'd']]; const flattened = arr.flat(); console.log(flattened); // => ["a", "b", "c", "d"]
之前,咱们常用reduce()
或concat()
来展平多维数组:
const arr = ['a', 'b', ['c', 'd']]; const flattend = [].concat.apply([], arr); // or // const flattened = [].concat(...arr); console.log(flattened); // => ["a", "b", "c", "d"]
请注意,若是提供的数组中有空值,它们会被丢弃:
const arr = ['a', , , 'b', ['c', 'd']]; const flattened = arr.flat(); console.log(flattened); // => ["a", "b", "c", "d"]
flat()
还接受一个可选参数,该参数指定嵌套数组应该被展平的级别数。 若是未提供参数,则将使用默认值1
:
const arr = [10, [20, [30]]]; console.log(arr.flat()); // => [10, 20, [30]] console.log(arr.flat(1)); // => [10, 20, [30]] console.log(arr.flat(2)); // => [10, 20, 30]
flatMap()
方法将map()
和flat()
组合成一个方法。 它首先使用提供的函数的返回值建立一个新数组,而后链接该数组的全部子数组元素。 来个例子:
const arr = [4.25, 19.99, 25.5]; console.log(arr.map(value => [Math.round(value)])); // => [[4], [20], [26]] console.log(arr.flatMap(value => [Math.round(value)])); // => [4, 20, 26]
数组将被展平的深度级别为1
.若是要从结果中删除项目,只需返回一个空数组:
const arr = [[7.1], [8.1], [9.1], [10.1], [11.1]]; // do not include items bigger than 9 arr.flatMap(value => { if (value >= 10) { return []; } else { return Math.round(value); } }); // returns: // => [7, 8, 9]
除了正在处理的当前元素外,回调函数还将接收元素的索引和对数组自己的引用。flat()
和flatMap()
方法目前处于第4阶段。
在建立Symbol时,能够为调试目的向其添加description
(描述)。有时候,可以直接访问代码中的description
是颇有用的。
ES2019 中为Symbol
对象添加了只读属性 description
,该对象返回包含Symbol
描述的字符串。
let sym = Symbol('foo'); console.log(sym.description); // => foo sym = Symbol(); console.log(sym.description); // => undefined // create a global symbol sym = Symbol.for('bar'); console.log(sym.description); // => bar
try catch
语句中的catch
有时候并无用,思考下面代码:
try { // 使用浏览器可能还没有实现的功能 } catch (unused) { // 这里回调函数中已经帮咱们处理好的错误 }
此代码中的catch
回调的信息并无用处。 但这样写是为了不SyntaxError
错误。 ES2019能够省略catch
周围的括号:
try { // ... } catch { // .... }
matchAll()
方法是ES2020 第4阶段提议,它针对正则表达式返回全部匹配(包括捕获组)的迭代器对象。
为了与match()
方法保持一致,TC39 选择了“matchAll”而不是其余建议的名称,例如 “matches” 或 Ruby的 “scan”。看个简单的例子:
const re = /(Dr\. )\w+/g; const str = 'Dr. Smith and Dr. Anderson'; const matches = str.matchAll(re); for (const match of matches) { console.log(match); } // logs: // => ["Dr. Smith", "Dr. ", index: 0, input: "Dr. Smith and Dr. Anderson", groups: undefined] // => ["Dr. Anderson", "Dr. ", index: 14, input: "Dr. Smith and Dr. Anderson", groups: undefined]
此正则表达式中的捕获组匹配字符“Dr”,后跟一个点和一个空格。\w+
匹配任何单词字符一次或屡次。 而且g
标志指示引擎在整个字符串中搜索模式。
以前,必须在循环中使用exec()
方法来实现相同的结果,这不是很是有效:
const re = /(Dr\.) \w+/g; const str = 'Dr. Smith and Dr. Anderson'; let matches; while ((matches = re.exec(str)) !== null) { console.log(matches); } // logs: // => ["Dr. Smith", "Dr.", index: 0, input: "Dr. Smith and Dr. Anderson", groups: undefined] // => ["Dr. Anderson", "Dr.", index: 14, input: "Dr. Smith and Dr. Anderson", groups: undefined]
重要的是要注意尽管match()
方法能够与全局标志g
一块儿使用来访问全部匹配,但它不提供匹配的捕获组或索引位置。 比较如下代码:
const re = /page (\d+)/g; const str = 'page 2 and page 10'; console.log(str.match(re)); // => ["page 2", "page 10"] console.log(...str.matchAll(re)); // => ["page 2", "2", index: 0, input: "page 2 and page 10", groups: undefined] // => ["page 10", "10", index: 11, input: "page 2 and page 10", groups: undefined]
在这篇文章中,咱们仔细研究了 ES2019 中引入的几个关键特性,包括Object.fromEntries()
,trimStart()
, trimEnd()
, flat()
, flatMap()
,symbol
对象的description 属性以及可选的catch
。
尽管一些浏览器尚未彻底实现这些特性,但可使用Babel和其余JavaScript转换器,仍然能够在项目中使用它们。
干货系列文章汇总以下,以为不错点个Star,欢迎 加群 互相学习。
https://github.com/qq44924588...
我是小智,公众号「大迁世界」做者,对前端技术保持学习爱好者。我会常常分享本身所学所看的干货,在进阶的路上,共勉!
关注公众号,后台回复福利,便可看到福利,你懂的。