ES6-解構賦值及原理

基礎語法

數組

// 基礎類型解構
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/Map/Set

// 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概念

Iterator是一種接口,爲各類不同的數據解構提供統一的訪問機制。任何數據解構只要有Iterator接口,就能經過遍歷操做,依次按順序處理數據結構內全部成員。ES6中的for of的語法至關於遍歷器,會在遍歷數據結構時,自動尋找Iterator接口。數組

Iterator做用:數據結構

  • 爲各類數據解構提供統一的訪問接口
  • 使得數據解構能按次序排列處理
  • 可使用ES6最新命令 for of進行遍歷
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和value
done - 若是迭代器已經通過了被迭代序列時爲 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對象是有其緣由的。對象

參考文章

阮一峯ECMAScript 6接口

Mozilla - for of

Mozilla - Iteration protocols

相關文章
相關標籤/搜索