ES6(二)之变量的解构赋值

解构赋值

1. 数组的解构赋值

基本用法
  • 左右俩边结构一致
  • 右边必须是合法值
  • 声明与赋值不能分开必须在一句话里
let arr = [1, 2, 3];

let a =  arr[0];
let b = arr[1];
let c = arr[2];

console.log(a, b, c);

解构赋值后:

let [a, b, c]=[1, 2, 3];
console.log(a, b, c);

let [{a, b}, [{n1, n2, n3}], num, str] = [{1,2}, [{3, 4, 5}], 6, "str"];

let {a, b} = {1, 2};这种写法不合理,右边不属于合法的值

默认值
  • 解构赋值允许指定默认值
let [foo = true] = [];
foo // true

let [x, y = 'b'] = ['a']; // x='a', y='b'
let [x, y = 'b'] = ['a', undefined]; // x='a', y='b'
  • ES6 内部使用严格相等运算符(===),判断一个位置是否有值,所以当一个数组成员严格等于undefined,默认值菜户生效
let [x = 1] = [undefined];
x // 1

let [x = 1] = [null];
x // null

举例理解:

let [x = 1, y = x] = [];     // x=1; y=1
let [x = 1, y = x] = [2];    // x=2; y=2
let [x = 1, y = x] = [1, 2]; // x=1; y=2
let [x = y, y = 1] = [];     // ReferenceError: y is not defined

2. 对象的解构赋值

使用
  1. 对象的解构与数组的解构有一个重要的不同,数组的元素是按次序排列的,变量的取值由他的位置决定的,而对象的属性没有次序,变量必须与属性同名,才能取到正确的值
let  {a, b} = {a: "aaa", b: "bbb"};
a //"aaa"
b //"bbb"

let { c } = {a: "aaa", b: "bbb"};
c //undefined
  1. 如果变量名与属性名不一致
let { foo: baz } = { foo: 'aaa', bar: 'bbb' };
baz // "aaa"

let obj = { first: 'hello', last: 'world' };
let { first: f, last: l } = obj;
f // 'hello'
l // 'world'

let { foo: baz } = { foo: 'aaa', bar: 'bbb' };
baz // "aaa"
foo // error: foo is not defined

上面的代码,foo是匹配的模式,baz才是变量。真正被赋值的是变量baz,而不是模式foo。
3. 解构用于嵌套结构的对象

let obj = {
  p: [
    'Hello',
    { y: 'World' }
  ]
};

let { p: [x, { y }] } = obj;
x // "Hello"
y // "World"

这个时候,p是模式,不是变量,因此不会被赋值,如果p要欧威变量赋值,应该写成如下:

let obj = {
  p: [
    'Hello',
    { y: 'World' }
  ]
};

let { p, p: [x, { y }] } = obj;
x // "Hello"
y // "World"
p // ["Hello", {y: "World"}]
默认值
var {x = 3} = {};
x // 3

var {x, y = 5} = {x: 1};
x // 1
y // 5

默认值生效的条件是,对象的属性值严格等于undefined。

注意点
  1. JS会将{x}理解成代码块 ,只有在不将大括号写在行首,才能解决这个问题
//错误的写法
let x;
{x} = {x: 1};

//正确的写法
({x} = {x:1}});
  1. 解构赋值允许等号左边的模式之中,不放置任何变量名。但是表达式毫无意义
({} = [true, false]);
  1. 由于数组本质是特殊的对象,因此可以对数组进行对象属性的解构。
let arr = [1, 2, 3];
let {0 : first, [arr.length - 1] : last} = arr;
first // 1
last // 3

3. 字符串的解构赋值

  1. 字符串也可以解构赋值,字符串被转换成了一个类似数组的对象
const [a, b, c, d, e] = 'hello';
a // "h"
b // "e"
c // "l"  
d // "l"
e // "o"
  1. 类似数组的对象都有length属性
let {length : len} = "hello";
len //5

4. 数值和布尔值的解构赋值

  1. 数值和布尔值的包装对象都有toString属性
let {toString : s} = 123;
s === Number.prototype.toString // true

let {toString: s} = true;
s === Boolean.prototype.toString // true
  1. undefinednull无法转为对象,所以对他们进行解构赋值,都会报错TypeError
let { prop :x } = undefined;  //TypeError
let { prop : y } = null; //TypeError

5. 函数参数的解构赋值

  1. 函数参数的解构赋值举例说明:
[[1, 2], [3, 4]].map(([a, b]) => a + b);
// [ 3, 7 ]
  1. 函数参数数据的解构的默认值,函数的参数是对象,通过对对象进行解构,得到xy值,解构失败等于xy默认值
function move({x = 0, y = 0} = {}) {
  return [x, y];
}

move({x: 3, y: 8}); // [3, 8]
move({x: 3}); // [3, 0]
move({}); // [0, 0]
move(); // [0, 0]
  1. 对函数的参数指定默认值会有不同的效果
function move({ x , y } = {x = 0, y = 0}) {
  return [x, y];
}

move({x: 3, y: 8}); // [3, 8]
move({x: 3}); // [3, undefined]
move({}); // [undefined, undefined]
move(); // [0, 0]

6. 圆括号问题

不能使用圆括号的情况
  1. 变量声明语句
//全部报错
let [(a)] = [1];

let {x: (c)} = {};
let ({x: c}) = {};
let {(x: c)} = {};
let {(x): c} = {};

let { o: ({ p: p }) } = { o: { p: 2 } };
  1. 函数参数
//报错
function f( [(a)] ){ return a; }
  1. 赋值语句的模式
//报错
({ p : a}) = { p : 42 };
可以使用圆括号的情况
  • 赋值语句的非模式部分,注意以下条件:
  1. 是赋值语句,而不是声明语句
  2. 圆括号都不属于模式的一部分
[(b)] = [3]; // 正确
({ p: (d) } = {}); // 正确
[(parseInt.prop)] = [3]; // 正确

在第二个例子里,p是模式,(d)是变量

7. 解构赋值的用处

  • 交换变量的值
let x =1;
let y = 2;
[ x, y ] = [ y, x ];
  • 从函数返回多个值
    • 函数只能返回一个值,只能将其返回数组或者对象中,使用解构赋值易于取值。
// 返回数组
 function example (){
 	return [1, 2, 3];
 }
 let [a, b, c]=example();

//返回对象
 function example(){
 	return {
 		foo: 1,
 		bar: 2
	 }
 }
 let { foo, bar }  = example();
  • 函数参数的定义
    • 方便参数与变量名的一一对应
//参数是有序的
function f([x, y, z]){...}
f([1, 2, 3]);

//参数是无序的
function f({x, y, z}) { ... }
f({z: 3, y: 2, x: 1});
  • 提取 JSON 数据
let jsonData = {
  id: 42,
  status: "OK",
  data: [867, 5309]
};

let { id, status, data: number } = jsonData;

console.log(id, status, number);
// 42, "OK", [867, 5309]
  • 函数参数的默认值
jQuery.ajax = function (url, {
  async = true,
  beforeSend = function () {},
  cache = true,
  complete = function () {},
  crossDomain = false,
  global = true,
  // ... more config
} = {}) {
  // ... do stuff
};
  • 遍历 Map 结构
const map = new Map();
map.set('first', 'hello');
map.set('second', 'world');

for (let [key, value] of map) {
  console.log(key + " is " + value);
}
// first is hello
// second is world

// 获取键名
for (let [key] of map) {
  // ...
}

// 获取键值
for (let [,value] of map) {
  // ...
}
  • 输入模块的指定方法
    • vue的按需导入组件Mint-UI
import { Header, Swipe, SwipeItem, Button, Lazyload } from 'mint-ui'
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值