// 基礎類型解構
let [a, b, c] = [1, 2, 3]
console.log(a, b, c) // 1, 2, 3
// 對象數組解構
let [a, b, c] = [{name: '1'}, {name: '2'}, {name: '3'}]
console.log(a, b, c) // {name: '1'}, {name: '2'}, {name: '3'}
// ...解構
let [head, ...tail] = [1, 2, 3, 4]
console.log(head, tail) // 1, [2, 3, 4]
// 嵌套解構
let [a, [b], d] = [1, [2, 3], 4]
console.log(a, b, d) // 1, 2, 4
// 解構不成功爲undefined
let [a, b, c] = [1]
console.log(a, b, c) // 1, undefined, undefined
// 解構默認賦值
let [a = 1, b = 2] = [3]
console.log(a, b) // 3, 2
複製代碼
// 對象屬性解構
let { f1, f2 } = { f1: 'test1', f2: 'test2' }
console.log(f1, f2) // test1, test2
// 能夠不按照順序,這是數組解構和對象解構的區別之一
let { f2, f1 } = { f1: 'test1', f2: 'test2' }
console.log(f1, f2) // test1, test2
// 解構對象重命名
let { f1: rename, f2 } = { f1: 'test1', f2: 'test2' }
console.log(rename, f2) // test1, test2
// 嵌套解構
let { f1: {f11}} = { f1: { f11: 'test11', f12: 'test12' } }
console.log(f11) // test11
// 默認值
let { f1 = 'test1', f2: rename = 'test2' } = { f1: 'current1', f2: 'current2'}
console.log(f1, rename) // current1, current2
複製代碼
// 參數解構
function func1({ x, y }) {
return x + y
}
func1({ x: 1, y: 2}) // 3
function func1({ x = 1, y = 2 }) {
return x + y
}
func1({x: 4}) // 6
複製代碼
// String
let [ a, b, c, ...rest ] = 'test123'
console.log(a, b, c, rest) // t, e, s, [ 't', '1', '2', '3' ]
// Map
let [a, b] = new Map().set('f1', 'test1').set('f2', 'test2')
console.log(a, b) // [ 'f1', 'test1' ], [ 'f2', 'test2' ]
// Set
let [a, b] = new Set([1, 2, 3])
console.log(a, b) // 1, 2
複製代碼
解構是ES6提供的語法糖,其實內在是針對可迭代對象
的Iterator接口
,經過遍歷器
按順序獲取對應的值進行賦值。這裏須要提早懂得ES6的兩個概念:es6
Iterator是一種接口,爲各類不同的數據解構提供統一的訪問機制。任何數據解構只要有Iterator接口,就能經過遍歷操做,依次按順序處理數據結構內全部成員。ES6中的for of的語法至關於遍歷器,會在遍歷數據結構時,自動尋找Iterator接口。數組
Iterator做用:數據結構
function makeIterator(array) {
var nextIndex = 0
return {
next: function() {
return nextIndex < array.length ?
{value: array[nextIndex++]} :
{done: true}
}
};
}
var it = makeIterator([0, 1, 2])
console.log(it.next().value) // 0
console.log(it.next().value) // 1
console.log(it.next().value) // 2
複製代碼
可迭代對象是Iterator接口的實現。這是ECMAScript 2015的補充,它不是內置或語法,而僅僅是協議
。任何遵循該協議點對象都能成爲可迭代對象。可迭代對象得有兩個協議:可迭代協議和迭代器協議。函數
可迭代協議
:對象必須實現@@iterator方法。即對象或其原型鏈上必須有一個名叫Symbol.iterator
的屬性。該屬性的值爲無參函數,函數返回迭代器協議。ui
屬性 | 值 |
---|---|
Symbol.iterator |
返回一個對象的無參函數,被返回對象符合迭代器協議。 |
迭代器協議
:定義了標準的方式來產生一個有限或無限序列值。其要求必須實現一個next()方法,該方法返回對象有done(boolean)和value屬性。spa
屬性 | 值 |
---|---|
next |
返回一個對象的無參函數,被返回對象擁有兩個屬性:done和valuedone - 若是迭代器已經通過了被迭代序列時爲 true。這時 value 可能描述了該迭代器的返回值。若是迭代器能夠產生序列中的下一個值,則爲 false。這等效於連同 done 屬性也不指定。value - 迭代器返回的任何 JavaScript 值。done 爲 true 時可省略。 |
經過以上可知,自定義數據結構,只要擁有Iterator接口,並將其部署到本身的Symbol.iterator屬性上,就能夠成爲可迭代對象,能被for of循環遍歷。rest
// 自定義可迭代對象
let obj = {
[Symbol.iterator] : function() {
return{
next: function() {
return { value: 1, done: true }
}
}
}
}
for (let item of obj) {
console.log(item) // 不會報錯,由於obj已是可迭代對象
}
複製代碼
String、Array、Map、Set等原生數據結構都是可迭代對象,能夠經過for of循環遍歷它。故能夠經過ES6解構語法糖依次獲取對應的值。code
// String
let str = 'test'
let iterFun = str[Symbol.iterator]
let iterator = str[Symbol.iterator]()
let first = iterator.next() // 等效於 let [first] = 'test'
console.log(iterFun, iterator, first)
// 打印
// [Function: [Symbol.iterator]], {}, { value: 't', done: false }
// Array
let arr = ['a', 'b', 'c'];
let iter = arr[Symbol.iterator]();
// 如下等效於 let [first, second, third, four] = ['a', 'b', 'c']
let first = iter.next() // { value: 'a', done: false }
let second = iter.next() // { value: 'b', done: false }
let third = iter.next() // { value: 'c', done: false }
let four = iter.next() // { value: undefined, done: true }
複製代碼
原生object對象是默認沒有部署Iterator接口,即object不是一個可迭代對象。由於遍歷時,不知道到底哪一個屬性先遍歷,哪一個屬性後遍歷,須要開發者手動指定。不過object部署Iterator接口沒有必要,由於ES6提供了Map數據結構。實際上對象被解構時,會被看成Map進行解構。因此雖然Map和Object不少地方類似,但ES6引入Map、Set對象是有其緣由的。對象