国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

《JavaScript高級程序設計》(第3版)讀書筆記 第11章 DOM拓展

luck / 410人閱讀

摘要:對的兩個主要拓展是選擇和。以下插入標記的拓展已經納入了規范。在寫模式下,會根據指定的字符串創建新的子樹,然后用這個子樹完全替換調用元素。在刪除帶有時間處理程序或引用了其他對象子樹時,就有可能導致內存占用問題。

盡管DOM作為API已經非常完善了,但為了實現更多功能,仍然會有一些標準或專有的拓展。2008年之前,瀏覽器中幾乎所有的拓展都是專有的,此后W3C著手將一些已經成為事實標準的專有拓展標準化,并寫入規范中。

對DOM的兩個主要拓展是Selectors API (選擇API)和HTML5。這兩個拓展都源自開發社區,而將某些常見做啊及API標準化,一直是眾望所歸。

選擇符API

jQuery的核心就是通過CSS選擇符查詢DOM文檔取得元素的引用,從而拋開了getElementById()gettElementByTagName()

Selectors API 是又W3C發起制定的一個標準,致力于讓瀏覽器原生支持CSS查詢。所有實現這一功能的JavaScript庫都會寫一個基礎的CSS解析器,然后再使用已有的DOM方法查詢文檔并找到匹配的節點。而把這個功能變成原生API之后,解析和樹查詢操作可以在瀏覽器內部通過編譯后的代碼完成,極大改善了性能。

querySelector()方法

querySelector()方法接收一個CSS選擇符,返回與該模式匹配的第一個元素,如果沒有找到匹配的元素,返回null

// 取得body元素
var body = document.querySelector("body");

// 取得ID為“myDiv”的元素
var myDiv = document.querySelector("#myDiv");

// 取得類為"selected"的第一個元素
var selected = document.querySelector(".selected");

// 取得類為"button"的第一個圖像元素
var selected = document.querySelector("img.button");

如果傳入了不支持的選擇符,會拋出錯誤。

querySelectorAll()方法

和上面類似,但返回的不僅僅是匹配的第一個元素,而是一個NodeList的實例,如果沒有匹配,NodeList就是空的。

// 取得某div中所有em元素
var ems = document.getElementById("myDiv").querySelectorAll("em");

// 取得類為“selected”的所有元素
var selected = document.querySelectorAll(".selected");

// 取得所有p元素中的所有strong元素
var strongs = document.querySelectorAll("p strong");

取得返回NodeList中的每一個元素,可以使用item()方法,也可以使用方括號語法。

matchesSelector()方法

Selector API Level2 規范為Element類型新增了一個matchesSelector()方法。接收一個參數,即CSS選擇符,如果調用元素與該選擇符匹配返回true,否則返回false。

截止2011年年中,還沒有瀏覽器支持該方法,不過有一些實驗性的實現。因此如果你想使用這個方法,最好編寫一個包裝函數:

function matchesSelector(element, selector) {
  if (element.matchesSelecotr) {
    return element.matchesSelecotr(selector)
  } else if (element.msMatchesSelecotr) {
    return element.msMatchesSelecotr(selector)
  } else if (element.mozMatchesSelecotr) {
    return element.mozMatchesSelecotr(selector)
  } else if (element.webkitMatchesSelecotr) {
    return element.webkitMatchesSelecotr(selector)
  } else {
    throw new Error("Not supported.");
  }
}
元素遍歷

對于元素間的空格,IE9及之前版本不會返回文本節點,而其他所有瀏覽器都會返回文本節點。為了彌補這一差異,而又同時保持DOM規范不變,Element Traversal 規范新定義了一組屬性。

childElementCount: 返回子元素(不包括文本節點和注釋)的個數

firstElementChild: 指向第一個子元素;firstChild的元素版

lastElementChild: 指向最后一個子元素;lastChild的元素版

previousElementSibling: 指向前一個同輩元素;previousSibling的元素版

nextElementSibling: 指向前一個同輩元素;nextSibling的元素版

// 跨瀏覽器遍歷某元素的所有子元素

// 老版的兼容性代碼
var i,
    len,
    child = element.firstChild;
while(child != element.lastChild) {
  // 檢查是不是元素
  if (child.nodeType == 1) {
    processChild(child);
  }
  child = child.nextSibling;
}

// 使用新版的方法
var i,
    len,
    child = element.firstChild;
while(child != element.lastElementChild) {
  processChild(child);
  child = child.nextElementSibling;
}

支持Element Traversal 規范的瀏覽器有 IE9+ Firefox3.5+ Safari4+ Chrome Opera10+

HTML5

HTML5所有之前的版本對JavaScript接口的描述不過三言兩語,主要篇幅都用于定義標記,與JavaScript相關的內容一概交由DOM規范去定義。

HTML5規范則圍繞如何使用新增標記定義了大量JavaScript API。其中一些API與DOM重疊,定義了瀏覽器應該支持的DOM拓展。

由于HTML5涉及的面非常廣,本節只討論與DOM節點相關的內容。

與類(class)相關的擴充

HTML5 新增了很多API,致力于簡化CSS類的用法。

getElementsByClassName()方法
// 取得所有類中包含"username"和"current"的元素
// 類名的先后順序無所謂
var allCurrentUsernames = docment.getElementsByClassName("username current");

// 取得ID為"myDiv"的元素中帶有類名"selected"的所有元素
var selected = document.getElementById("myDiv").getElemenstByClassName("selected");

支持getElementsByClassName()方法的瀏覽器 IE9+ Firefox3+ Safari3.1+ Chrome Opera9.5+

classList屬性

在操作類名時,需要通過className屬性添加、刪除和替換類名。因為className中是一個字符串,所以即使只是修改字符串一部分,也必須每次都設置整個字符串的值。

HTML5 新增了一種操作類名的方式,可以讓操作更簡單也更安全,那就是為所有元素添加了classList屬性。

classList屬性是新集合類型 DOMTokenList的實例。與其他DOM集合類似,DOMTokenList 有一個表示自己包含多少元素的length屬性,而要去的每個元素可以使用item()方法,或者方括號語法。此外,這個新類型還定義如下方法:

add(value): 將給定的字符串值添加到列表中。如果值已經存在,就不添加了。

contains(value): 表示列表中是否存在給定的值,如果存在返回true,反之false。

remove(value): 從列表中刪除給定的字符串。

toggle(value): 如果列表中已經存在給定的值,刪除它;如果沒有,添加它。

有了classList屬性,除非你需要刪除所有類名,或者完全重寫元素的class屬性,否則就用不到className屬性了。

支持classList的瀏覽器 Firefox3.6+ Chrome

焦點管理

HTML5添加了輔助DOM焦點的功能。

document.activeElement屬性,始終會引用DOM中當前獲得了焦點的元素。元素獲得焦點的方式有頁面加載、用戶輸入(通常是通過Tab鍵)和在代碼中調用focus()方法。

var button = document.getElementById("myButton");
button.focus();
console.log(document.activeElement === button);     // true

默認情況下,文檔剛剛加載完成時,document.activeElement中保存的是document.body元素的引用。文檔加載期間,document.activeElement的值為null

document.hasFocus()方法,這個方法用于確定文檔是否獲得了焦點。

var button = document.getElementById("myButton");
button.focus();
console.log(document.hasFocus());        // true

支持的瀏覽器 IE4+ Firefox3+ Safari4+ Chrome Opera8+

HTMLDocument的變化 readyState屬性

Document.readyState屬性有兩個可能的值:

loading正在加載文檔

complete已經加載完文檔

支持的瀏覽器 IE4+ Firefox3.6+ Safari Chrome Opera9+

if (document.readyState == "complete") {
  ...
}
兼容模式

自從IE6開始區分渲染頁面的模式是標準的還是混雜的,檢測頁面的兼容性就成為瀏覽器的必要功能。IE為此給document添加了一個名為compatMode的屬性,告訴開發人員瀏覽器采用了哪種渲染模式。

document.compatMode標準模式下等于"CSS1Compat",混雜模式下等于"BackCompat"。

最終HTML5將這個屬性納入標準

支持的瀏覽器 IE Firefox Safari3.1 Chrome Opera

if (document.compatMode == "CSS1Compat") {
  console.log("Standards mode");
} else {
  console.log("Quirks mode");
}
head屬性

HTML5新增了document.head屬性,與docuemnt.body對應

支持的瀏覽器 Chrome Safari 5+

var head = document.head || document.getElementsByTagName("head")[0];
字符集屬性

HTML5新增了幾個與文檔字符集有關的屬性

charset屬性表示文檔中實際使用的字符集,也可以用來指定新字符集。默認值是"UTF-16",可以通過元素、響應頭部或直接設置charset屬性修改這個值。

支持的瀏覽器 IE Safari Opera Chrome 。 Firefox 支持 document.Characterset

console.log(document.charset);     // "UTF-16"
document.charset = "UTF-8";

defaultCharset表示根據默認瀏覽器及操作系統的設置,當前文檔默認的字符集應該是什么。如果文檔沒有使用默認的字符集,那charsetdefaultCharset屬性值可能會不一樣。

if (document.charset != document.defaultCharset) {
  console.log("Custom character set being used.");
}

支持的瀏覽器 IE Safari Chrome 。

自定義數據屬性

HTML5 規定可以為元素添加非標準的屬性,但要添加前綴data- ,目的是為元素提供與渲染無關的信息,或者提供語義信息。這些屬性可以任意添加、隨便明明,只要以data-開頭即可。

添加了自定義屬性之后,可以通過元素的dataset屬性來訪問自定義屬性的值。dataset屬性的值時DOMStringMap的實例,也就是一個名值對的映射。在這個映射中,每個data-name形式的屬性都會有一個對應的屬性,只不過屬性沒有data-前綴。

var div = document.getElementById("myDiv");

// 取得自定義屬性的值
var appId = div.dataset.appId;
var myName = div.dataset.myname;

// 設置值
div.dataset.appId = 2345;
div.dataset.myname = "Michael";

// 有沒有"myname"值呢?
if (div.dataset.myname) {
  console.log("Hello, " + div.dataset.myname);
}
插入標記

雖然DOM為操作節點提供了細致入微的控制手段,但在需要給文檔插入大量HTML標記的情況下,通過DOM操作仍然非常麻煩,因為不僅要創建一系列DOM節點,還要小心按照正確的順序把它們連接起來。

相對而言,直接插入HTML字符串不僅更簡單,速度也更快。以下插入標記的DOM拓展已經納入了HTML5規范。

innerHTML屬性

在讀模式下,innerHTML屬性返回與調用元素的所有子節點(包括元素、注釋和文本節點)對應的HTML標記。

在寫模式下,innerHTML會根據指定的值創建新的DOM樹,然后用這個DOM樹完全替換調用元素原先的所有子節點。

This is a paragraph with a list following it.

  • Item 1
  • Item 2
  • Item 3

This is a paragraph with a list following it.

  • Item 1
  • Item 2
  • Item 3

不同瀏覽器返回的文本格式不會有所不同。IE和Opera會將所有標簽轉換為大小寫形式,而Safari、Chrome、Firefox則會原原本本的按照原先文檔中(或指定這些標簽時)的格式返回HTML,包括空格和縮進。

在寫模式下,傳入innerHTML的值都會按照瀏覽器處理HTML的標注方式轉換為元素(同樣因瀏覽器而異)。如果設置的值僅是純文本而沒有HTML標簽,那么結果就是設置純文本。

div.innerHTML = "Hello world!";
div.innerHTML = "Hello & welcom, "reader"!";
// 以上操作得到: 
Hello & welcome, "reader"!

使用innerHTML屬性也有一些限制,大多數瀏覽器中通過innerHTML插入

閱讀需要支付1元查看
<