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

資訊專欄INFORMATION COLUMN

通過DOM API 查找節(jié)點

shengguo / 2226人閱讀

摘要:我們可以通過來檢查某個節(jié)點是否有子節(jié)點。通過可以用來取得父元素,返回值可能會是一個元素節(jié)點根節(jié)點或節(jié)點。與類似,通過可以取得同層之間的下一個節(jié)點,如果已經(jīng)是最后一個節(jié)點,則返回。

在上一篇的分享當中,我們簡單介紹了BOM 與DOM,也了解到JavaScript 是怎么通過它們提供的方法來與瀏覽器進行溝通。

當一個網(wǎng)頁被載入到瀏覽器時,瀏覽器會首先分析這個HTML 文檔,然后會依照這份HTML 的內(nèi)容解析成DOM (Document Object Model,即文件對象模型)。

而DOM 是W3C 制定的一個規(guī)范,它是獨立于平臺與語言的標準。換言之,只要遵守這樣的規(guī)范,不管是什么平臺或者是什么語言開發(fā),都可以通過DOM 提供的API 來操作DOM 的內(nèi)容、結(jié)構(gòu)與樣式。

所以說,DOM 是網(wǎng)頁的根本,懂得控制DOM就可以控制整個網(wǎng)頁,做出良好的互動體驗。

那么在今天的分享中,我們就繼續(xù)來介紹DOM API 查找節(jié)點的方法吧。

前言:script標簽放哪里有區(qū)別嗎?

針對

接著,我們試著把

咦?怎么什么都沒有呢?而且也沒有錯誤信息,JavaScript真的如大家說的一樣,很垃圾嗎?

冷靜一下,容我解釋一下。

前面說過,當一個網(wǎng)頁被載入到瀏覽器時,瀏覽器會先分析這個HTML 文檔,由上而下依序來讀取解析:

所以上面jsbin例子中,當瀏覽器在 ... 之間遇到

上一篇文章說過,document對象是DOM tree的根節(jié)點,所以當我們要存取HTML時,都從document對象開始。而DOM的節(jié)點類型除了HTML元素節(jié)點(element nodes)外,還有文字節(jié)點(text nodes)、注釋節(jié)點(comment nodes)等。

而常見的DOM 選取方法有下列這些:

// 根據(jù)傳入的值,找到 DOM 中 id 為 "xxx" 的元素。
document.getElementById("xxx");

// 針對給定的 tag 名稱,返回所有符合條件的 NodeList 對象(節(jié)點的集合)
document.getElementsByTagName("xxx");

// 針對給定的 class 名稱,返回所有符合條件的節(jié)點集合
document.getElementsByClassName("xxx");

// 針對給定的 Selector 條件,返回第一個 或 所有符合條件的節(jié)點集合
document.querySelector("xxx"); 
document.querySelectorAll("xxx");
DOM 節(jié)點的類型

DOM 常用的節(jié)點類型有下面幾種:

可以通過節(jié)點類型常數(shù)或是對應數(shù)值來判斷:

document.nodeType === Node.DOCUMENT_NODE;   //true
document.nodeType === 9;   //true

其他不常用或是已經(jīng)廢棄的部分可以參考:MDN Node.nodeType一節(jié)。

DOM 節(jié)點間的查找遍歷(Traversing)

由于DOM 節(jié)點有分層的概念,于是節(jié)點與節(jié)點之間的關系,我們大致上可以分成以下兩種:

父子關系:除了document之外,每一個節(jié)點都會有個上層的節(jié)點,我們通常稱之為「父節(jié)點」 (Parent node),而相對地,從屬于自己下層的節(jié)點,就會稱為「子節(jié)點」 (Child node)。

兄弟關系:有同一個「父節(jié)點」的節(jié)點,那么他們彼此之間就是「兄弟節(jié)點」(Siblings node)。

而隔層的節(jié)點基本上沒有直接關系。

上圖中水平方向的鄰層節(jié)點為父子關系,垂直方向的同層節(jié)點為兄弟關系。

Node.childNodes

所有的DOM節(jié)點對象都有childNodes屬性,且此種屬性無法修改。

我們可以通過Node.hasChildNodes()來檢查某個DOM節(jié)點是否有子節(jié)點。

var node = document.querySelector("#hello");

// 如果 node 內(nèi)有子元素
if( node.hasChildNodes() ) {
    
    // 可以通過 node.childNodes[n] (n 為數(shù)字索引) 取得對應的節(jié)點
    // 注意,NodeList 對象內(nèi)容為即時更新的集合
    for (var i = 0; i < node.childNodes[i].length; i++) {
       // ...     
    }; 
}

Node.childNodes返回的可能會有這幾種:

HTML 元素節(jié)點(element nodes)

文字節(jié)點(text nodes),包含空格

注釋節(jié)點(comment nodes)

Node.firstChild

Node.firstChild可以取得Node節(jié)點的第一個子節(jié)點,如果沒有子節(jié)點則返回null。

要注意的是,子節(jié)點包括空白節(jié)點,如下面例子:

<p>
  <span>span 1span>
  <span>span 2span>
  <span>span 3span>
p>

<script>
  var p = document.querySelector("p");    
  
  // tagName 屬性可以取得 node 的標簽名稱
  console.log(p.firstChild.tagName);      // undefined
script>

因為取得的是

與第一個中間的換行字元,所以p.firstChild.tagName會得到undefined。所以改成這樣:

<p><span>span 1span><span>span 2span><span>span 3span>p>

<script>
  var p = document.querySelector("p");    

  // tagName 屬性可以取得 node 的標簽名稱
  console.log(p.firstChild.tagName);      // "SPAN"
script>

把中間的換行與空白移除,就會得到預期中的"SPAN"了。

Node.lastChild

Node.lastChild可以取得Node節(jié)點的最后一個子節(jié)點,如果沒有子節(jié)點則返回null。

Node.firstChild一樣的是,子節(jié)點也包括空白節(jié)點,所以像這樣:

<p>
  <span>span 1span>
  <span>span 2span>
  <span>span 3span>
p>

<script>
  var p = document.querySelector("p");    
  
  // textContent 屬性可以取得節(jié)點內(nèi)的文字內(nèi)容
  console.log(p.lastChild.textContent);      // "" (換行字元)
script>

得到的會是一個換行字元的空字符串。

移除節(jié)點之間多余的空白后:

<p><span>span 1span><span>span 2span><span>span 3span>p>

<script>
  var p = document.querySelector("p");    
  // textContent 屬性可以取得節(jié)點內(nèi)的文字內(nèi)容
  console.log(p.lastChild.textContent);      // "span 3"
script>

輸出的就會是正確的"span 3" 啦。

Node.parentNode

那么相較于Child系列,parentNode就單純一些。

通過Node.parentNode可以用來取得父元素,返回值可能會是一個元素節(jié)點(Element node)、根節(jié)點(Document node)或DocumentFragment節(jié)點。

<p><span>span 1span><span>span 2span><span>span 3span>p>

<script>
  var el = document.querySelector("span");   
  console.log( el.parentNode.nodeName );    // "P"
script>

Node.previousSibling

看完了DOM父與子之后,接著來看看兄弟節(jié)點。

通過Node.previousSibling可以取得同層之間的前一個節(jié)點,如果node已經(jīng)是第一個節(jié)點且前面無節(jié)點,則返回null

<p><span>span 1span><span>span 2span><span>span 3span>p>

<script>
  var el = document.querySelector("span");   
  console.log( el.previousSibling );    // null

  // document.querySelectorAll 會取得所有符合條件的集合,
  // 而 document.querySelectorAll("span")[2] 指的是「第三個」符合條件的元素。
  var el2 = document.querySelectorAll("span")[2];   
  console.log( el2.previousSibling.textContent );    // "span 2"
script>

Node.nextSibling

Node.previousSibling類似,通過Node.nextSibling可以取得同層之間的下一個節(jié)點,如果node已經(jīng)是最后一個節(jié)點,則返回null。

<p><span>span 1span><span>span 2span><span>span 3span>p>

<script>

  // document.querySelector 會取得第一個符合條件的元素
  var el = document.querySelector("span");

  console.log( el.nextSibling.textContent );    // "span 2"
script>

document.getElementsBy*與document.querySelector/ document.querySelectorAll的差異

今天分享了很多關于DOM的選取以及查找遍歷的方式,其中,像是document.getElementById以及document.querySelector因為取得的一定只會有一個元素/節(jié)點,所以不會有indexlength屬性。

document.getElementsBy**(注意,這里有個s)以及document.querySelectorAll則分別返回HTMLCollection與NodeList。

這兩者其實是有點差別的,HTMLCollection只收集HTML element 節(jié)點,而NodeList除了HTML element 節(jié)點,也包含文字節(jié)點、屬性節(jié)點等。當然兩者也有類似的地方,雖然不能使用數(shù)組的method,但這兩種都可以用數(shù)組索引的方式來存取內(nèi)容,也就是偽數(shù)組。

另一個需要注意的地方是,HTMLCollection/NodeList在大部分情況下是即時更新的,但通過document.querySelector/document.querySelectorAll取得的NodeList是靜態(tài)的。

啥意思呢?舉個例子:

<div id="outer">
  <div id="inner">innerdiv>
div>

<script>

  // 
var outerDiv = document.getElementById("outer"); // 所有的
標簽 var allDivs = document.getElementsByTagName("div"); console.log(allDivs.length); // 2 // 清空
下的節(jié)點 outerDiv.innerHTML = ""; // 因為清空了
下的節(jié)點,所以只剩下 outer console.log(allDivs.length); // 1 script>

如果改成document.querySelector的寫法:

<div id="outer">
  <div id="inner">innerdiv>
div>

<script>

  // 
var outerDiv = document.getElementById("outer"); // 所有的
標簽 var allDivs = document.querySelectorAll("div"); console.log(allDivs.length); // 2 // 清空
下的節(jié)點 outerDiv.innerHTML = ""; // document.querySelector 返回的是靜態(tài)的 NodeList,不受 outerDiv 更新影響 console.log(allDivs.length); // 2 script>

那么以上就是今天所要介紹的內(nèi)容啦。

在后續(xù)的文章會再繼續(xù)說明DOM API新增/刪除/修改節(jié)點的部分,歡迎持續(xù)關注。

如果覺得文章對你有些許幫助,歡迎在我的GitHub博客點贊和關注,感激不盡!

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

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

相關文章

  • DOM 小總結(jié)

    摘要:在瀏覽器中,對象是繼承自類型的一個實例,表示整個頁面。它這種情況稱為文檔碎片,還無法顯示在瀏覽器的畫面中。創(chuàng)建一個文本節(jié)點并添加到新元素中的代碼如下總結(jié)文檔對象模型,是針對和文檔的一個應用程序編程接口,描繪了一個層次化的節(jié)點樹。 DOM 是什么 文檔對象模型,是針對 HTML 和 XML 文檔的一個 API (應用程序編程接口), 描繪了一個層次化的節(jié)點樹。 D: document 當...

    hyuan 評論0 收藏0
  • DOM 小總結(jié)

    摘要:在瀏覽器中,對象是繼承自類型的一個實例,表示整個頁面。它這種情況稱為文檔碎片,還無法顯示在瀏覽器的畫面中。創(chuàng)建一個文本節(jié)點并添加到新元素中的代碼如下總結(jié)文檔對象模型,是針對和文檔的一個應用程序編程接口,描繪了一個層次化的節(jié)點樹。 DOM 是什么 文檔對象模型,是針對 HTML 和 XML 文檔的一個 API (應用程序編程接口), 描繪了一個層次化的節(jié)點樹。 D: document 當...

    yzd 評論0 收藏0
  • 重學前端學習筆記(二十三)--狹義的文檔對象DOM

    摘要:事件觸發(fā)和監(jiān)聽事件相關。文檔是一個由標簽嵌套而成的樹形結(jié)構(gòu),因此,也是使用樹形的對象模型來描述一個文檔。節(jié)點的寫法三是樹繼承關系的根節(jié)點。七表示一個上的范圍,這個范圍是以文字為最小單位的。 筆記說明 重學前端是程劭非(winter)【前手機淘寶前端負責人】在極客時間開的一個專欄,每天10分鐘,重構(gòu)你的前端知識體系,筆者主要整理學習過程的一些要點筆記以及感悟,完整的可以加入winter的...

    Guakin_Huang 評論0 收藏0
  • 重學前端學習筆記(二十三)--狹義的文檔對象DOM

    摘要:事件觸發(fā)和監(jiān)聽事件相關。文檔是一個由標簽嵌套而成的樹形結(jié)構(gòu),因此,也是使用樹形的對象模型來描述一個文檔。節(jié)點的寫法三是樹繼承關系的根節(jié)點。七表示一個上的范圍,這個范圍是以文字為最小單位的。 筆記說明 重學前端是程劭非(winter)【前手機淘寶前端負責人】在極客時間開的一個專欄,每天10分鐘,重構(gòu)你的前端知識體系,筆者主要整理學習過程的一些要點筆記以及感悟,完整的可以加入winter的...

    wuaiqiu 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<