摘要:一個(gè)對(duì)象如果要有可被循環(huán)調(diào)用的接口,就必須在的屬性上部署遍歷器生成方法原型鏈上的對(duì)象具有該方法也可。阮大神案例上面代碼是一個(gè)類部署接口的寫法。屬性對(duì)應(yīng)一個(gè)函數(shù),執(zhí)行后返回當(dāng)前對(duì)象的遍歷器對(duì)象。
最近看阮一峰阮大神的ES6,剛剛看到Iterator和for...of循環(huán)這一章,小作筆記跟大家略微分享一下,不足之處還望大家多多指正
Iterator(遍歷器)就是一種機(jī)制;任何數(shù)據(jù)結(jié)構(gòu)只要是部署了iterator接口,就可以完成遍歷操作(即依次處理該數(shù)據(jù)的所有成員);
Iterator的作用有三個(gè):一是為各種數(shù)據(jù)結(jié)構(gòu),提供一個(gè)統(tǒng)一的、簡(jiǎn)便的訪問(wèn)接口;二是使得數(shù)據(jù)結(jié)構(gòu)的成員能夠按某種次序排列;三是ES6創(chuàng)造了一種新的遍歷命令for...of循環(huán),Iterator接口主要供for...of消費(fèi)。(阮大神原話);
只要是一個(gè)對(duì)象部署了Symbol.interator接口,就可以用for...of遍歷該對(duì)象,同時(shí)也可以調(diào)用該接口的Symbol.interator方法調(diào)用next()方法對(duì)對(duì)象進(jìn)行遍歷,不同的是for..of是對(duì)該對(duì)象的值的輸出,而next()返回的是對(duì)象。
例如下面的例子:
如果調(diào)用next方法就是
調(diào)用next()方法會(huì)返回一個(gè)對(duì)象{value:value;done:true or false};直到對(duì)遍歷對(duì)象的值便利完成之后,next()方法會(huì)返回一個(gè){value:undefined; done:true}代表已經(jīng)遍歷完成,再調(diào)用next()方法也不會(huì)報(bào)錯(cuò)但是依舊會(huì)返回{value:undefined; done:true};
有一些對(duì)象是原生就封裝好的Symbol.iterator接口,可以直接調(diào)用,當(dāng)然也可以直接用for...of遍歷;
在ES6中,有三類數(shù)據(jù)結(jié)構(gòu)原生具備Iterator接口:數(shù)組、某些類似數(shù)組的對(duì)象、Set和Map結(jié)構(gòu)。
如果沒(méi)有原生的Symbol.iterator接口,想用for...of遍歷,就需要自己在該對(duì)象中部署Symbol.iterator接口,例如
一個(gè)對(duì)象如果要有可被for...of循環(huán)調(diào)用的Iterator接口,就必須在Symbol.iterator的屬性上部署遍歷器生成方法(原型鏈上的對(duì)象具有該方法也可)。
一個(gè)對(duì)象如果要有可被for...of循環(huán)調(diào)用的Iterator接口,就必須在Symbol.iterator的屬性上部署遍歷器生成方法(原型鏈上的對(duì)象具有該方法也可)。 class RangeIterator { constructor(start, stop) { this.value = start; this.stop = stop; } [Symbol.iterator]() { return this; } next() { var value = this.value; if (value < this.stop) { this.value++; return {done: false, value: value}; } else { return {done: true, value: undefined}; } } } function range(start, stop) { return new RangeIterator(start, stop); } for (var value of range(0, 3)) { console.log(value); }//阮大神案例
上面代碼是一個(gè)類部署Iterator接口的寫法。Symbol.iterator屬性對(duì)應(yīng)一個(gè)函數(shù),執(zhí)行后返回當(dāng)前對(duì)象的遍歷器對(duì)象。
原型鏈上部署Symbol.iterator接口
function Obj(value){ this.value = value; this.next = null; } Obj.prototype[Symbol.iterator] = function(){ var iterator = { next: next }; var current = this; function next(){ if (current){ var value = current.value; var done = current == null; current = current.next; return { done: done, value: value } } else { return { done: true } } } return iterator; } var one = new Obj(1); var two = new Obj(2); var three = new Obj(3); one.next = two; two.next = three; for (var i of one){ console.log(i) } // 1 // 2 // 3
對(duì)象內(nèi)部部署
let obj = { data: [ "hello", "world" ], [Symbol.iterator]() { const self = this; let index = 0; return { next() { if (index < self.data.length) { return { value: self.data[index++], done: false }; } else { return { value: undefined, done: true }; } } }; } };
也就是說(shuō)如果一個(gè)對(duì)象沒(méi)有Symbol.iterator接口,可以在類本身上面部署,也可以在原型連上部署,也可以在對(duì)象內(nèi)部部署
對(duì)于類似數(shù)組的對(duì)象(存在數(shù)值鍵名和length屬性),部署Iterator接口,有一個(gè)簡(jiǎn)便方法,就是Symbol.iterator方法直接引用數(shù)組的Iterator接口。
NodeList.prototype[Symbol.iterator] = Array.prototype[Symbol.iterator]; // 或者 NodeList.prototype[Symbol.iterator] = [][Symbol.iterator]; [...document.querySelectorAll("div")] // 可以執(zhí)行了
具體請(qǐng)看例子
let iterable = { 0: "a", 1: "b", 2: "c", length: 3, [Symbol.iterator]: Array.prototype[Symbol.iterator] }; for (let item of iterable) { console.log(item); // "a", "b", "c" }
**但是請(qǐng)記住,這個(gè)方法僅僅適用于類數(shù)組對(duì)象,上面的也可以直接用Array.from(iterable)轉(zhuǎn)換成數(shù)組來(lái)遍歷,例如
let arrayLike = { length: 2, 0: "a", 1: "b" }; for (let x of Array.from(arrayLike)) { console.log(x); }//a b
對(duì)于普通對(duì)象這兩個(gè)方法是不管用的,**例如
let iterable = { edition: "a", writer: "b", read: "c", length: 3, }; for (let item of Array.from(iterable)) { console.log(item); }
上面的代碼就會(huì)輸出三個(gè)undefined
let iterable = { edition: "a", writer: "b", read: "c", length: 3, [Symbol.iterator]: Array.prototype[Symbol.iterator] }; for (let item of iterable) { console.log(item); //報(bào)錯(cuò) }
上面的代碼就會(huì)拋異常的;
其實(shí)ES6很多地方都用的到Iterator這個(gè)接口,例如:
解構(gòu)賦值
擴(kuò)展運(yùn)算符
yield*
最后需要說(shuō)下,字符串既是類數(shù)組對(duì)象同時(shí)自己也有原生Symbol.iterator接口可以直接調(diào)用,例如
var str = "hell";
for(let v of str){
console.log(v);//"h","e","l","l"
}
對(duì)于字符串來(lái)說(shuō),for...of循環(huán)還有一個(gè)特點(diǎn),就是會(huì)正確識(shí)別32位UTF-16字符。
for...of 區(qū)別于for循環(huán),for循環(huán)比較麻煩,但是是最原始的方法;
for...of 區(qū)別于數(shù)組的forEach方法,因?yàn)閒orEach方法是從頭到尾執(zhí)行,不會(huì)跳出,但是遇到break或者return,continue會(huì)跳出循環(huán)
有著同for...in一樣的簡(jiǎn)潔語(yǔ)法,但是沒(méi)有for...in那些缺點(diǎn)。
不同用于forEach方法,它可以與break、continue和return配合使用。
提供了遍歷所有數(shù)據(jù)結(jié)構(gòu)的統(tǒng)一操作接口。
更詳細(xì)的請(qǐng)查看阮大神(http://es6.ruanyifeng.com/#docs/iterator)本章內(nèi)容
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/86086.html
摘要:本文從使用對(duì)數(shù)組進(jìn)行遍歷開(kāi)始說(shuō)起,粗略對(duì)比使用進(jìn)行遍歷的差異,并由此引入中可迭代對(duì)象迭代器的概念,并對(duì)其進(jìn)行粗略介紹。說(shuō)到這里,就繼續(xù)說(shuō)一下迭代器關(guān)閉的情況了。確實(shí),符合可迭代協(xié)議和迭代器協(xié)議的。 本文從使用 forEach 對(duì)數(shù)組進(jìn)行遍歷開(kāi)始說(shuō)起,粗略對(duì)比使用 forEach , for...in , for...of 進(jìn)行遍歷的差異,并由此引入 ES6 中 可迭代對(duì)象/迭代器 的概...
摘要:語(yǔ)法參數(shù)當(dāng)前遍歷項(xiàng)。遍歷的范圍在第一次調(diào)用前就會(huì)確定。已刪除的項(xiàng)不會(huì)被遍歷到。的是由提出的,目的是作為遍歷所有數(shù)據(jù)結(jié)構(gòu)的統(tǒng)一方法。不僅可以遍歷數(shù)組,還可以遍歷結(jié)構(gòu),某些類似數(shù)組的對(duì)象如對(duì)象對(duì)象,對(duì)象,以及字符串。 即使是最簡(jiǎn)單的循環(huán),其中也深藏學(xué)問(wèn) ECMAScript5(es5)有三種for循環(huán) 簡(jiǎn)單for for in forEach ECMAScript6(es6)新增 fo...
摘要:中可以實(shí)現(xiàn)遍歷的數(shù)據(jù)類型主要是對(duì)象,其中包括普通對(duì)象與數(shù)組。遍歷器是一種接口,為各種不同的數(shù)據(jù)結(jié)構(gòu)提供統(tǒng)一的訪問(wèn)機(jī)制。實(shí)例五遍歷器對(duì)象實(shí)例五是的遍歷過(guò)程,通過(guò)手動(dòng)調(diào)用其對(duì)象的方法實(shí)現(xiàn)信息獲取。為每個(gè)數(shù)組元素執(zhí)行函數(shù)。 前言 ??將依據(jù)自身痛點(diǎn)學(xué)習(xí),計(jì)劃對(duì)原生JavaScript寫一個(gè)系統(tǒng),本文為第一篇,感興趣的同學(xué)可以關(guān)注個(gè)人公眾號(hào):ZeroToOneMe,或者github博客,將持續(xù)...
摘要:有兩個(gè)協(xié)議可迭代協(xié)議和迭代器協(xié)議。為了變成可迭代對(duì)象,一個(gè)對(duì)象必須實(shí)現(xiàn)或者它原型鏈的某個(gè)對(duì)象必須有一個(gè)名字是的屬性迭代器協(xié)議該迭代器協(xié)議定義了一種標(biāo)準(zhǔn)的方式來(lái)產(chǎn)生一個(gè)有限或無(wú)限序列的值。 ECMAScript 2015的幾個(gè)補(bǔ)充,并不是新的內(nèi)置或語(yǔ)法,而是協(xié)議。這些協(xié)議可以被任何遵循某些約定的對(duì)象來(lái)實(shí)現(xiàn)。有兩個(gè)協(xié)議:可迭代協(xié)議和迭代器協(xié)議。 可迭代協(xié)議 可迭代協(xié)議允許 JavaScri...
摘要:我關(guān)注的賀老賀師俊前輩最近發(fā)表個(gè)這樣一條微博雖然這條微博沒(méi)有引起大范圍的關(guān)注和討論,但是作為新人,我陷入了思考。通過(guò)賀老的微博,對(duì)一個(gè)問(wèn)題進(jìn)行探究,最終找到核心成員的一文,進(jìn)行參考并翻譯。 我關(guān)注的賀老—賀師俊前輩@johnhax 最近發(fā)表個(gè)這樣一條微博: showImg(https://segmentfault.com/img/remote/1460000010452807); 雖然...
閱讀 2685·2021-11-18 10:02
閱讀 3416·2021-09-28 09:35
閱讀 2595·2021-09-22 15:12
閱讀 754·2021-09-22 15:08
閱讀 3117·2021-09-07 09:58
閱讀 3476·2021-08-23 09:42
閱讀 737·2019-08-30 12:53
閱讀 2086·2019-08-29 13:51