深入理解ES6之迭代器與生成器

迭代器

迭代器是被設(shè)計專用于迭代的對象,帶有特定接口关噪。所有的迭代器對象都擁有next()方法拿穴,會返回一個結(jié)果對象。該結(jié)果對象有兩個屬性:對應(yīng)下一個值的value,以及一個布爾類型的done秀仲,其值為true時表示沒有更多值可供使用融痛。迭代器持有一個指向集合位置的內(nèi)部指針,每當(dāng)調(diào)用了next()方法神僵,迭代器就會返回相應(yīng)的下一個值雁刷。

使用es5模擬一個迭代器

function createIterator(items) {
  var i = 0
  // 返回一個迭代器對象
  return {
    next: () => {
      var done = i >= items.length
      var value = !done ? items[++i] : undefined
      return {
        done,
        value
      }
    }
  }
}

var list = createIterator([1, 2, 3, 4, 5])
console.log(list.next())    // { done: false, value: 2 }
console.log(list.next())    // { done: false, value: 3 }
console.log(list.next())    // { done: false, value: 4 }
console.log(list.next())    // { done: false, value: 5 }
console.log(list.next())    // { done: true, value: undefined }

生成器

生成器(generator)是能返回一個迭代器的函數(shù)。生成器函數(shù)由放在function關(guān)鍵字之后的一個星號(*)來表示保礼,并能使用新的yield關(guān)鍵字沛励。將星號緊跟在function關(guān)鍵字之后,或是在中間留出空格炮障,都是沒問題的侯勉。

function *createIterator() {
  yield 1
  yield 2
  yield 3
}

let iterator = createIterator()

console.log(iterator.next().value)    // 1
console.log(iterator.next().value)    // 2
console.log(iterator.next().value)    // 3

可迭代對象與 for-of 循環(huán)

與迭代器緊密相關(guān)的是,可迭代對象(iterable)是包含Symbol.iterator屬性的對象铝阐。這個Symbol.iterator知名符號定義了為指定對象返回迭代器的函數(shù)址貌。在 ES6 中,所有的集合對象(數(shù)組徘键、 Set 與 Map )以及字符串都是可迭代對象练对,因此它們都被指定了默認(rèn)的迭代器。

for-of循環(huán)在循環(huán)每次執(zhí)行時會調(diào)用可迭代對象的next()方法吹害,并將結(jié)果對象的value值存儲在一個變量上螟凭。循環(huán)過程會持續(xù)到結(jié)果對象的done屬性變成true為止。

let value = [1, 2, 3]
for (let num of value) {
  console.log(num)
}

輸出:

1
2
3

訪問默認(rèn)迭代器 Symbol.iterator

let values = [1, 2, 3, 4, 5]
let iterator = values[Symbol.iterator]()
iterator.next()    // {value: 1, done: false}
iterator.next()    // {value: 2, done: false}
iterator.next()    // {value: 3, done: false}

集合的迭代器

ES6 具有三種集合對象類型:數(shù)組它呀、 Map 與 Set 螺男。

  • entries():返回一個包含鍵值對的迭代器;
  • values():返回一個包含集合中的值的迭代器纵穿;
  • keys():返回一個包含集合中的鍵的迭代器下隧。
let map = new Map([['name', 'age'], ['map', 5]])
map.entries()    // MapIterator {"name" => "age", "iterator" => 5, "friends" => "none"}
map.values()    // MapIterator {"map", 5}
map.keys()    // MapIterator {"name", "age"}

let set = new Set([1, 2, 3, 4, 5])
set.entries()    // SetIterator {1 => 1, 2 => 2, 3 => 3}
set.values()    // SetIterator {1, 2, 3, 4, 5}
set.keys()    // SetIterator {1, 2, 3, 4, 5}

let arr = [1, 2, 3, 4, 5]
arr.entries()    // Array Iterator {}
arr.values()    // Array Iterator {}
arr.keys()    // Array Iterator {}

調(diào)用迭代器方法返回的數(shù)據(jù)需要通過 for...of... 來遍歷

字符串的迭代器

從 ES5 發(fā)布開始, JS 的字符串就慢慢變得越來越像數(shù)組谓媒。例如 ES5 標(biāo)準(zhǔn)化了字符串的方括號表示法淆院,用于訪問其中的字符(即:使用text[0]來獲取第一個字符,以此類推)句惯。

所以也可以像遍歷集合那樣遍歷字符串

  • 傳統(tǒng) for 循環(huán)
  • for...of...

NodeList 的迭代器

文檔對象模型( DOM )具有一種NodeList類型土辩,用于表示頁面文檔中元素的集合。NodeList對象與數(shù)組都使用了length屬性來標(biāo)明項的數(shù)量抢野,并且都使用方括號表示法來訪問各個項拷淘。然而本質(zhì)上來說,NodeList與數(shù)組的行為是完全不同的指孤,這會引發(fā)許多混亂启涯。
著默認(rèn)迭代器被附加到 ES6 , DOM 關(guān)于NodeList的規(guī)定也包含了一個默認(rèn)迭代器(此規(guī)定在 HTML 規(guī)范而非 ES6 規(guī)范中),其表現(xiàn)方式與數(shù)組的默認(rèn)迭代器一致逝嚎。這意味著你可以將NodeList用于for-of循環(huán)扁瓢,或用于其他使用對象默認(rèn)迭代器的場合。

let divs = document.getElementsByTagName("div")
for (let div of divs) {
  console.log(div.id)
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末补君,一起剝皮案震驚了整個濱河市引几,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌挽铁,老刑警劉巖伟桅,帶你破解...
    沈念sama閱讀 211,194評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異叽掘,居然都是意外死亡楣铁,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評論 2 385
  • 文/潘曉璐 我一進(jìn)店門更扁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來盖腕,“玉大人,你說我怎么就攤上這事浓镜±A校” “怎么了?”我有些...
    開封第一講書人閱讀 156,780評論 0 346
  • 文/不壞的土叔 我叫張陵膛薛,是天一觀的道長听隐。 經(jīng)常有香客問我,道長哄啄,這世上最難降的妖魔是什么雅任? 我笑而不...
    開封第一講書人閱讀 56,388評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮咨跌,結(jié)果婚禮上沪么,老公的妹妹穿的比我還像新娘。我一直安慰自己虑润,他們只是感情好成玫,可當(dāng)我...
    茶點故事閱讀 65,430評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著拳喻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪猪腕。 梳的紋絲不亂的頭發(fā)上冗澈,一...
    開封第一講書人閱讀 49,764評論 1 290
  • 那天,我揣著相機與錄音陋葡,去河邊找鬼亚亲。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的捌归。 我是一名探鬼主播肛响,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼惜索!你這毒婦竟也來了特笋?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤巾兆,失蹤者是張志新(化名)和其女友劉穎猎物,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體角塑,經(jīng)...
    沈念sama閱讀 44,122評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡蔫磨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,459評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了圃伶。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片堤如。...
    茶點故事閱讀 38,605評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖窒朋,靈堂內(nèi)的尸體忽然破棺而出搀罢,到底是詐尸還是另有隱情,我是刑警寧澤炼邀,帶...
    沈念sama閱讀 34,270評論 4 329
  • 正文 年R本政府宣布魄揉,位于F島的核電站,受9級特大地震影響拭宁,放射性物質(zhì)發(fā)生泄漏洛退。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,867評論 3 312
  • 文/蒙蒙 一杰标、第九天 我趴在偏房一處隱蔽的房頂上張望兵怯。 院中可真熱鬧,春花似錦腔剂、人聲如沸媒区。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽袜漩。三九已至,卻和暖如春湾碎,著一層夾襖步出監(jiān)牢的瞬間宙攻,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評論 1 265
  • 我被黑心中介騙來泰國打工介褥, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留座掘,地道東北人递惋。 一個月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像溢陪,于是被迫代替她去往敵國和親萍虽。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,472評論 2 348