继续第二章,变量的解构赋值。
什么是解构赋值
ES6 允许按照一定的模式,从数组和对象中提取值,这种方式被称为解构(Destructuring)。
数组的解构赋值
基本用法
以前为变量赋值,只能直接指定值,例如:
1 | let a = 1; |
但是解构赋值允许这样写:
1 | let [a, b, c] = [1, 2, 3]; |
意义就跟上面一样了,为 a
,b
,c
分别赋值。
本质上,这种写法属于“模式匹配”,只要等号两边的模式相同,左边的变量就会被赋予相应的值。以下是更多例子:
1 | let [foo, [[bar], baz]] = [1, [[2], 3]]; |
如果解构不成功,变量的值就等于 undefined
。
另一种情况是不完全解构,即等号左边的模式,只匹一部分的等号右边的数组。这种情况下,解构依然可以成功。
1 | let [x, y] = [1, 2, 3]; |
如果等号的右边不是数组(或者严格的说,不是可遍历的结构),那么将会报错。
默认值
解构赋值允许指定默认值。但要注意的是,ES6 内部使用严格相等运算符号(===
),判断一个位置是否有值。所以,只有当一个数组成员严格等于 undefined
,默认值才会生效。
1 | let [foo = true] = []; |
如果默认值是一个表达式,那么这个表达式是惰性求值的,即只有在用到的时候,才会求值。
1 | function f() { |
对象的解构赋值
对象的解构赋值和数组有一个重要的不同,数组的元素是按次序排列的,变量的取值由它的位置决定;而对象的属性没有次序,变量必须与属性同名,才能取到正确的值。
1 | let {foo, bar} = {foo: 'aaa', bar: 'bbb'}; |
如果变量名与属性名不一致,必须写成下面这样:
1 | let {foo: baz} = {foo: 'aaa', bar: 'bbb'}; |
实际上说明,对像的解构赋值是下面形式的简写:
1 | let {foo: foo, bar: bar} = {foo: 'aaa', bar: 'bbb'}; |
也就是说,对象的解构赋值的内部机制,是先找到同名属性,然后再赋给对应的变量。真正被赋值的是后者,而不是前者。
与数组一样,解构也可以用于嵌套结构的对象。
1 | let obj = { |
在上面的例子中,p
是模式,不是变量,所以 p
不会被赋值,如果 p
也要作为变量赋值,要写成下面这样:
1 | // 续上 |
默认值
同数组解构的默认值一样,对象的属性值严格等于 undefined
时默认值才会生效。如果解构失败,变量的值等于 undefined
。
1 | var {x = 3} = {}; |
如果解构模式是嵌套的对象,而且子对象所在的父属性不存在,那么将会报错。如果要将一个已经声明的变量用于解构赋值,也要小心。
1 | let {foo: {bar}} = {baz: 'baz'}; |
上面的写法会报错,因为 JS 引擎会将 {x}
解释成一个代码块,从而发生语法错误。只有不将花括号写在行首,避免 JS 将它解释成代码块,才能解决问题。
1 | let x; |
解构赋值允许等号左边的模式之中,不放置任何变量名,因为可以写出非常古怪的表达式,它们没有意义,但是语法合法,可以执行。
1 | ({} = [true, false]); |
对象的解构赋值,可以很方便的将现有对象的方法,赋值到某个变量。
1 | let {log, sin, cos} = Math; |
由于数组的本质也是对象,因此可以对数组进行对象的解构。
1 | let arr = [1, 2, 3]; |
字符串的解构赋值
对字符串进行解构赋值时,字符串被转换成了一个类似数组的对象。
1 | const [a, b, c, d, e] = 'hello'; |
因为字符串还有个 length
属性,所以这个属性也可被解构。
1 | let {length: len} = 'hello'; |
数值和布尔值的解构赋值
如果等号右边是数值和布尔值,则会先转为对象。
1 | let {toString: s} = 123; |
解构赋值的规则是,只要等号右边的值不是对象或数组,就先将其转为对象。由于 undefined
和 null
无法转换成对象,所以对它们进行解构的时候,会报错。
1 | let {prop: x} = undefined; |
函数参数的解构赋值
函数的参数也可以使用解构赋值。
1 | function add([x, y]) { |
不过,如果用下面的写法,会得到不一样的结果:
1 | function move({x, y} = {x: 0, y: 0}) { |
上面这段代码是为函数 move
的参数指定默认值,而不是为 x
和 y
变量指定默认值,所以会得到不同结果。
undefined
就会触发函数参数的默认值。
1 | [1, undefined, 3].map((x = 'yes') => x); |
圆括号的问题
解构的过程中,如果模式中出现了圆括号,可能会引起处理上的歧义。ES6 的规则是,只要有可能导致解构的歧义,就不得使用圆括号。
不能使用圆括号的情况
变量声明语句
1 | // 全部报错 |
函数参数
函数参数也属于变量声明,因此不能带有圆括号。
1 | // 全部报错 |
赋值语句的模式
1 | // 全部报错 |
可以使用圆括号的情况
可以使用圆括号的情况只有一种:赋值语句的非模式部分,可以使用圆括号。
1 | // 以下均正确 |
除了上面注释里写明的原因,上面三行都能正确执行的原因,是它们都是赋值语句,而不是声明语句。
用途
交换变量的值
1 | let x = 1; |
从函数返回多个值
函数只有一个返回值,如果要返回多个值,只能将它们放在数组或者对象里返回。然后用解构赋值取出它们。
1 | // 返回一个数组 |
函数参数的定义
解构赋值可以方便的将一组参数与变量名对应起来。
1 | // 参数是一组有序的值 |
提取 JSON 数据
1 | let jsonData = { |
函数参数默认值
1 | jQuery.ajax = function (url, { |
在 Vuex 中的应用
创建带命名空间的辅助函数:
1 | import {createNamespacedHelpers} from 'vuex'; |
然后添加组件定义:
1 | export default { |
重命名带命名空间的辅助函数是个很好的实践,因为未来可能还会为其他模块添加辅助函数。例如,如果不这么做,可能最后会有两个 mapGetters
,而这是不可行的。这里将 mapGetters
重命名为 postsGetters
,同时将 mapActions
重命名为 postsActions
。
参考资料
本文基本是上面链接的笔记和再整理。