在我的上一篇隨筆里面分析了jQuery的構造函數,jQuery對象中有一個原型方法init才是是真正的構造函數,通過init的原型對象跟jQuery的原型對象保持引用關系使得init的實例可以正常調用jQuery的原型方法,就好像是jQuery的實例一樣。下面就來看看init這個幕后的構造函數是怎么寫的:
init: function( selector, context, rootjQuery ) {...}
可以看到這個方法接受3個參數,其前兩個參數是jQuery方法傳遞過來的
var jQuery = function( selector, context ) {// The jQuery object is actually just the init constructor 'enhanced'return new jQuery.fn.init( selector, context, rootjQuery );},
Selector原則上可以輸入任意值,但并不是所有值都是有意義的,只有undefined、DOM 元素、字符串、函數、jQuery 對象、普通 JavaScript 對象這幾種類型是有效的,這個參數是通常是填寫的但是不填寫也不會報錯
Context作為執行上下文或者叫執行范圍可以不傳入,或者傳入 DOM 元素、jQuery 對象、普通 JavaScript 對象之一
參數 rootjQuery:包含了 document 對象的 jQuery 對象,用于 document.getElementById() 查找失敗、selector 是選擇器表達式且未指定 context、selector 是函數的情況,其實就是$(document)。
下面根據參數的不同分為12種情況逐個討論
1.selector 可以轉換為false
// Handle $(""), $(null), or $(undefined)if ( !selector ) {return this;}
源碼中的注釋已經寫得很清楚了,當是這三種情況時直接return不進行任何處理
2.參數 selector 是 DOM 元素
例如: $(document)這種寫法
// Handle $(DOMElement)if ( selector.nodeType ) {this.context = this[0] = selector;this.length = 1;return this;}
只要是dom元素肯定有節點類型,然后把這個節點變成jquery對象的第一個元素并且賦值給上下文context,length屬性是jQuery的原型屬性默認為0
// The default length of a jQuery object is 0
length: 0,
這里有了一個元素之后就把length屬性修改為1。return this 操作使得函數執行后的結果依然是jQuery對象這樣就可以實現類似$(document).each()這樣的鏈式調用了。最終得到的類似這樣的{0:document,context:document,length:1....}對象,其實所有的情況最后都會變成這種形式的對象,除了jQuery原型屬性和方法之外就是獲取的dom節點并且按照阿拉伯數字依次排列,所以我們可以使用$(selector)[0]的形式代替$(selector).get(0)來獲取dom對象。例如:
<!doctype html><html> <head> <title></title> </head> <body> <div></div> <div></div> <div></div> </body> <script src='jquery-1.7.1.js'></script> <script> console.log($('div'));/*[div, div, div, prevObject: jQuery.fn.jQuery.init[1], context: document, selector: "div", constructor: function, init: function…]0: div1: div2: divcontext: documentlength: 3prevObject: jQuery.fn.jQuery.init[1]__proto__: jQuery[0]selector: "div".*/ </script></html>
3.參數是特殊的字符串“body”
由于body元素在一個文檔對象中只有一個所以單獨列出來處理
// The body element only exists once, optimize finding itif ( selector === "body" && !context && document.body ) {this.context = document;this[0] = document.body;this.selector = selector;this.length = 1;return this;}
這里有3個條件必須同時滿足,第二個必須沒有上下文的條件我也不是太理解,$(‘body',document)這樣的看起來很正常的寫法也會被這種情況“忽視”
console.log($('body',document)); /* jQuery.fn.jQuery.init[1]0: bodycontext: documentlength: 1prevObject: jQuery.fn.jQuery.init[1]selector: "body"__proto__: jQuery[0]*/
雖然和$('body')的結果是一樣的,但是卻被當做兩種情況來看待,可能是因為body只有一個上下文只能是document沒有必要添加吧,否則又要判斷上下文是不是document。第三個條件是保證document.body必須存在,那么什么情況下會前兩個情況滿足document.body又不存在呢?首先就是js代碼先于html代碼加載時會出現這個是初學者經常會犯的錯誤,通常我們要寫成:
$(function(){...})
或者
$(document).ready(function(){...})
其實這兩個是一樣的調取的是一個方法,dom加載這一塊以后在分析。對此我們可以做個測試html代碼如下:
<!doctype html> <html> <head> <title></title> <script src='jquery-1.7.1.js'></script> <script> $('body') </script> </head> <body> <div></div> <div></div> <div></div> </body> </html>
然后再jQuery源代碼里面輸出selector、context和document.body
console.log(selector+context+document.body);// The body element only exists once, optimize finding itif ( selector === "body" && !context && document.body ) {this.context = document;this[0] = document.body;this.selector = selector;this.length = 1;return this;}
雖然我們只寫了一個其實執行了四次,只有最后一次才是是我們調用后的結果,最后一次的結果是bodyundefinednull這個時候前兩個就是滿足的但是最后一個是null?;叵肫鸬谝黄猨Query總體架構架構里面undefined會被重新,那么document.body會不會被重寫為null呢?當我嘗試在代碼中修改時就會報錯看來是不會的,那這個條件就是預防沒有加載html就執行的情況吧
第四種是除了上述的字符串情況之外的其他字符串,情況比較多放在下一篇吧。
以上所述就是本文的全部內容了,希望大家能夠喜歡。
新聞熱點
疑難解答