成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

遍歷DOM元素的children屬性遇到的坑

weknow619 / 1768人閱讀

摘要:小結(jié)這下我們可以得出結(jié)論了個(gè)屬性返回的對(duì)象不止能遍歷到子元素,還能遍歷到來自其原型的三個(gè)屬性。既要防止那些添加修改了原型屬性的對(duì)象遍歷出多余的的結(jié)果,也要防止類似這種非標(biāo)準(zhǔn)屬性返回一個(gè)屬性的枚舉性不可控的對(duì)象的坑。

問題的引出

關(guān)于DOM元素的children屬性,以前我只在意它和childNodes屬性的區(qū)別:即children屬性只會(huì)返回子元素節(jié)點(diǎn)集合,而childNodes返回的就不止元素節(jié)點(diǎn),還有文本節(jié)點(diǎn)等所有子節(jié)點(diǎn)集合。這樣看來,children似乎是我們獲取子元素而舍棄其他類型的子節(jié)點(diǎn)的最佳選擇,雖然說在IE8-的瀏覽器下用它還會(huì)返回注釋節(jié)點(diǎn),但兼容起來也是很簡(jiǎn)單的。

我們知道,children返回的子元素集合實(shí)際上是一個(gè)類似數(shù)組的HTMLCollection對(duì)象,那接下來我們要獲取每個(gè)子元素自然要遍歷它咯,但是一遍歷,問題就出來了:

    

上面的代碼使用了for-in進(jìn)行遍歷,但我們預(yù)料中的結(jié)果并未出現(xiàn),以chrome為例,運(yùn)行結(jié)果是這個(gè):

這就是我在前面的原生js練習(xí)題-第一課那篇文章中提到過的坑了,其中有兩個(gè)奇怪的問題:

多返回了length等幾個(gè)在數(shù)組應(yīng)該是不可枚舉的屬性。

把有id的元素重復(fù)了兩次。

關(guān)于問題1

先說一下,上面提到的第一點(diǎn)在各個(gè)瀏覽器里情況都相同,而第二點(diǎn)關(guān)于返回的元素是否重復(fù)在各瀏覽器下情況還不同。

我們先討論第一點(diǎn),這里要考慮for-in循環(huán)遍歷對(duì)象時(shí)的規(guī)則比較奇葩:對(duì)象自身和繼承到的可枚舉屬性都會(huì)被遍歷到。所以為確定多遍歷到的內(nèi)容到底是自身還是原型上的屬性,我們來驗(yàn)證一下:

    console.log(Object.keys(o)); //["0","1","2","i","ii"]
    console.log(Object.getOwnPropertyNames(o)); //["0","1","2","i","ii"]

Object.keys()方法返回的是可枚舉的自身屬性的屬性名組成的數(shù)組,而Object.getOwnPropertyNames()返回的是所有自身屬性的屬性名組成的數(shù)組(含可枚舉和不可枚舉)。在這里我們沒有看到length、item()、namedItem()三個(gè)屬性的身影,由此斷定他們不是HTMLCollection對(duì)象自身的屬性,但既然能被for-in遍歷到那就只能是來自HTMLCollection原型的可枚舉屬性。我們可以用Object.getOwnPropertyDescriptor()來驗(yàn)證其在原型上的可枚舉性:

    console.log(Object.getOwnPropertyDescriptor(o.__proto__, "length").enumerable); //true
    console.log(Object.getOwnPropertyDescriptor(o.__proto__, "item").enumerable); //true
    console.log(Object.getOwnPropertyDescriptor(o.__proto__, "namedItem").enumerable); //true
關(guān)于問題2

解決了多出來的三個(gè)屬性的來源,我們?cè)倩剡^頭看看為什么會(huì)把有id的元素重復(fù)了兩次。觀察用Object.keys()方法返回的數(shù)組,這兩次一次用下標(biāo)做屬性名、一次用id名作屬性名。但其實(shí)兩個(gè)屬性名指向的是同一個(gè)對(duì)象:

    o[0]===o["i"] //true
    o[1]===o["ii"] //true

可見之所以for-in會(huì)把id的元素重復(fù)遍歷兩次,不是因?yàn)橛衖d的元素都添加進(jìn)HTMLCollection對(duì)象兩次,只是一個(gè)元素有了兩個(gè)屬性名而已,這是chrome的情況(我的版本是48.0.2564.116 m),但放到火狐和IE下結(jié)果卻還有點(diǎn)所不同:

    //FF
    console.log(Object.keys(o)); //["0", "1", "2"]
    console.log(Object.getOwnPropertyNames(o)); // ["0", "1", "2", "i", "ii"]
    o[0]===o["i"] //true
    o[1]===o["ii"] //true

    //IE11
    console.log(Object.keys(o)); //["i", "ii", "2"]
    console.log(Object.getOwnPropertyNames(o)); // ["i", "ii", "2"]
    o[0]===o["i"] //true
    o[1]===o["ii"] //true

可見雖然不同的瀏覽器返回的HTMLCollection對(duì)象都存在有id的子元素有兩個(gè)屬性名的情況,但從Object.keys(o)的結(jié)果看,火狐和IE對(duì)同一元素默認(rèn)只取一個(gè)屬性名。所以如果你在火狐或IE運(yùn)行一開始那段代碼,就會(huì)發(fā)現(xiàn)for-in遍歷時(shí)火狐和IE也只對(duì)同一元素訪問一次,不會(huì)像chrome那樣重復(fù)遍歷。然而我們還看到,這兩個(gè)瀏覽器間選取的屬性名也不同,火狐優(yōu)先選擇下標(biāo)形式,而IE優(yōu)先選擇id形式,同時(shí)由Object.getOwnPropertyNames(o)的結(jié)果我們還可以窺探出火狐實(shí)現(xiàn)選取屬性名的機(jī)制可能是通過將id形式的屬性名設(shè)為不可枚舉來實(shí)現(xiàn)的,至于IE就不清楚了。

小結(jié)

這下我們可以得出結(jié)論了:children個(gè)屬性返回的HTMLCollection對(duì)象不止能遍歷到子元素,還能遍歷到來自其原型的length、item()、namedItem()三個(gè)屬性。而且一旦遍歷到的子元素有id,就存在HTMLCollection對(duì)象里一個(gè)元素會(huì)有兩個(gè)屬性名的問題,更讓人蛋疼的是各瀏覽器對(duì)這兩個(gè)屬性名的選取各不相同。當(dāng)然,最根本原因還是因?yàn)?b>children屬性現(xiàn)在還沒被正式納入標(biāo)準(zhǔn),在使用這種非標(biāo)準(zhǔn)屬性時(shí)我們難免遇到一些奇葩的狀況。

所以這也告誡我們,如果對(duì)一個(gè)非標(biāo)準(zhǔn)屬性的特點(diǎn)不是特別了解,還是不要輕易使用它,否則出現(xiàn)的問題往往是你難以控制的。但如果你還是覺得children使用起來方便,那在使用時(shí)就得謹(jǐn)記這些問題,比如在遍歷子元素時(shí)最好放棄for-in循環(huán),老老實(shí)實(shí)使用基本的for循環(huán)去遍歷數(shù)字索引吧,這樣就和遍歷數(shù)組差不多,不會(huì)遍歷到那些多出來的屬性了。

而至于for-in,最好只用來遍歷數(shù)組或簡(jiǎn)單的對(duì)象。既要防止那些添加、修改了原型屬性的對(duì)象遍歷出多余的的結(jié)果,也要防止類似children這種非標(biāo)準(zhǔn)屬性返回一個(gè)屬性的枚舉性不可控的對(duì)象的坑。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/87704.html

相關(guān)文章

  • 自己實(shí)現(xiàn)一個(gè)簡(jiǎn)單虛擬 DOM

    摘要:直到內(nèi)部的全部循環(huán)結(jié)束為止,才進(jìn)入下一個(gè)元素,當(dāng)循環(huán)結(jié)束時(shí),內(nèi)部的節(jié)點(diǎn)都已經(jīng)生成好了。 自己實(shí)現(xiàn)虛擬 DOM 從 HTML 中提煉數(shù)據(jù)結(jié)構(gòu) 先來看下我們的 HTML 傅雷家書 讀家書,想付雷 從 HTML 中我們可以抽離出它的數(shù)據(jù)結(jié)構(gòu): 首先頁面中只需要一個(gè)根節(jié)點(diǎn)root,定義為:nodesDate數(shù)組 root內(nèi)有兩個(gè)子元素h1和span,數(shù)組有兩項(xiàng),每項(xiàng)為內(nèi)...

    luffyZh 評(píng)論0 收藏0
  • XPath 是一個(gè)好工具

    摘要:一個(gè)表達(dá)式是由一個(gè)或多個(gè)被分割的定位步組成。對(duì)于此類斷言,我們可以使用謂詞根據(jù)額外的遍歷樹來過濾出符合條件的節(jié)點(diǎn)。所以用來做一些低水平或與應(yīng)用無關(guān)的事情遍歷樹來找指定屬性的節(jié)點(diǎn)讓人蛋疼。這是一個(gè)專門用來讓你使用簡(jiǎn)潔的慣用表達(dá)式來遍歷的工具。 編者注: XPath 即為XML路徑語言(XML Path Language),它是一種用來確定XML文檔中某部分位置的語言。 XPat...

    Ilikewhite 評(píng)論0 收藏0
  • XPath 是一個(gè)好工具

    摘要:一個(gè)表達(dá)式是由一個(gè)或多個(gè)被分割的定位步組成。對(duì)于此類斷言,我們可以使用謂詞根據(jù)額外的遍歷樹來過濾出符合條件的節(jié)點(diǎn)。所以用來做一些低水平或與應(yīng)用無關(guān)的事情遍歷樹來找指定屬性的節(jié)點(diǎn)讓人蛋疼。這是一個(gè)專門用來讓你使用簡(jiǎn)潔的慣用表達(dá)式來遍歷的工具。 編者注: XPath 即為XML路徑語言(XML Path Language),它是一種用來確定XML文檔中某部分位置的語言。 XPat...

    codecraft 評(píng)論0 收藏0
  • 詳解NodeList 和 HTMLCollection 和 Array

    摘要:展開的屬性后發(fā)現(xiàn),繼承于一個(gè)對(duì)象,而這個(gè)對(duì)象又繼承于對(duì)象。這證實(shí)了我們對(duì)的猜想。是比較新的模型,相比更加完善,不光有元素,還有節(jié)點(diǎn)和。關(guān)于,和的關(guān)系,就是長得像,有個(gè)別相似的功能,但是是完全不一樣的東西。 Array,NodeList, HTMLCollection這三個(gè)概念和它們之間的關(guān)系有很多做了幾年前端的同學(xué)都搞不清楚,經(jīng)常遇到但是又感覺很陌生,剪不斷理還亂的感覺。今天咱們就來理...

    xietao3 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<