jQuery源代碼深入解析

jQuery源代碼深入解析,很多人覺得jquery、ext等一些開源js源代碼 十分的晦澀,讀不懂,遇到問題需要調試也很費勁。其實我個人感覺主要是有幾個方面的原因:

對一些js不常用的語法、操作符不熟悉
某個function中又嵌套瞭一些內部的function,使得整個代碼的層次結構不像java代碼那麼清晰。
js中允許變量先使用後定義,會造成我們看代碼時候忽然冒出來一個變量、function,卻找不到是在哪裡定義的。

那麼今天給大傢分享一下我的經驗,掃清你的障礙。

一些晦澀的操作符:

(function(){})();

幾乎所有的開源js代碼開篇都是這樣(function(……){……})(……);

下面是Jquery的部分源碼:

(function( window, undefined ) {
    var jQuery = function( selector, context ) {
        // The jQuery object is actually just the init constructor 'enhanced'
        return new jQuery.fn.init( selector, context );
    },

    // Map over jQuery in case of overwrite
    _jQuery = window.jQuery,

    // Map over the $ in case of overwrite
    _$ = window.$,
    ……
    indexOf = Array.prototype.indexOf;

    // Expose jQuery to the global object
    window.jQuery = window.$ = jQuery;
})(window);

那麼這個操作符(function(){})();到底是什麼意思呢?

(function(){})中的定義瞭一個function,緊接著的()表示立即執行這個function。

我們看到Jquery源碼第一個()中是定義瞭一個匿名function( window, undefined ) {};接著末尾有個(window),就表示執行這個匿名function,並傳入參數window。

在匿名function( window, undefined ) {}中,定義瞭一個局部變量jQuery;然後在末尾我們看到Jquery末尾有一句window.jQuery = window.$ = jQuery;這句代碼就表示,將此前定義的jQuery導出到window對象。這也是為什麼我們可以在代碼任何地方直接使用$、jQuery對象,因為在這裡已經將$、jQuery對象掛載到window下去瞭,而window.$、window.jQuery與直接使用$、jQuery是沒有區別的。

(註意,這個window對象是傳入的參數window,而不是瀏覽器window對象!!一個形參、一個實參。我們可以在定義function的時候,將參數window取名為其他字符。所以我們看到jquery.min.js中這個匿名function變成瞭(function(E,B){})(window);)

通常(function(){})()用來封裝一些私有成員或者公共成員的導出。

令人迷惑的","

我們知道,一般用於一次定義多個變量、定義多個參數等。像上面的jQuery源碼中在var jQuery後面,使用,一次定義瞭很多個變量。 但是,像下面的代碼,可能大傢就不一定看得懂瞭:

//html:

jQuery(document).ready(function() { var showName=function(){ var value,nameInput=$("#nameHide"); return nameInput.show(),value=nameInput.val(); }; alert(showName()); }); //結果:彈出king

這裡的nameInput.show(),value=nameInput.val()中的,運算符的作用是返回,右側表達式的值。所以,return後面如果有多個表達式,且表達式之間由,隔開,整個return表達式返回的是最後一個,右側的表達式的值。

,在開源代碼中常常被用於return表達式中,以及跟下面我們要講到的()運算符一起使用。

()廣義上的代碼包裝

我們遇到復雜的邏輯表達式時,我們通常會把需要一起運算的表達式用“()”包起來:(a||b)&&(c||d)

其實,我們可以這樣理解:()運算符將一個表達式包裹起來作為一個整體進行運算,然後返回這個整體的值。

那麼上面的(function(){})()中左側定義function的()也是這個作用,將這個function給包裹起來,然後返回這個function。我們調用方法一般是a();那麼(function(){})的作用就是返回這個function對象,然後(function(){})()右側的()表示調用這個function。

我們再來看其他的用法:

//html:

輸入錯誤!

jQuery(document).ready(function() { var nameValidate=function(){ var value,nameInput=$("#name"),nameErrorTip=$("#nameErrorTip"); return (value=nameInput.val(),value=="king")?(nameErrorTip.hide(),"對瞭,輸入為king!"):(nameErrorTip.show(),"請輸入king!"); }; alert(nameValidate()); }); //結果 nameErrorTip顯示,彈出"請輸入king!" //html:

輸入錯誤!

//結果 nameErrorTip隱藏,彈出"對瞭,輸入為king!"

這裡(value=nameInput.val(),value=="king")中()將裡面的表達式作為一個整體進行運算,而裡面的表達式又是由,構成的多個表達式組,所以執行的時候會把這多個表達式都執行一次,並且返回最後一個表達式的值!

所以(value=nameInput.val(),value=="king")執行時,先運算value的值,再判斷是否為"king"。如果為king,會執行(nameErrorTip.hide(),"對瞭,輸入為king!")。這個表達式又先將nameErrorTip隱藏,再返回一個"對瞭,輸入為king!"字符串作為 整個return的值。

||、&&、if()邏輯讓人頭暈

||、&&兩側參與運算的是邏輯表達式,if()中也是。但是我們在很多開源代碼中看到的||、&&參與運算的表達式看起來卻好像不是邏輯表達式……

下面節選一段jQuery.tool中的一段源碼:

e.circular || (f.onBeforeSeek(function(a, b) {
    setTimeout(function() {
        a.isDefaultPrevented()||(
            n.toggleClass(
                e.disabledClass,
                b <= 0
            ),
            o.toggleClass(
                e.disabledClass,
                b >= f.getSize() - 1
            )
        )
    }, 1)
}),
e.initialIndex || n.addClass(e.disabledClass)),
f.getSize() < 2 && n.add(o).addClass(e.disabledClass),
e.mousewheel && a.fn.mousewheel && b.mousewheel(function(a, b) {
    if (e.mousewheel) {
        f.move(b < 0 ? 1 : -1, e.wheelSpeed || 50);
        return !1
    }
});

這裡有多處||、&&。但與運算的表達式卻是調用某個函數的返回值。

其實,js中的邏輯表達式是按照真值、假值來分的。true是真值;1是真值;一個對象也是真值;false是假值;""、0是假值。

在js中&&、||不一定都是用來判斷一個表達式的邏輯值是true、false,更多的是用來依據真值或者假值執行相應操作!

我們知道,||運算的時候,會先運算左側的表達式的值,如果為真值,那麼真個表達式就為真值,而同時右側表達式是真值、假值都不重要,因為右側表達式都不再繼續參與運算瞭。又如果左側為假值,則繼續運算右側表達式。

&&則先運算左側表達式,兩側表達式,一個為假值,則整個表達式為假值。

這裡關鍵是這個真值或者假值的運算過程中,我們可以使用上面介紹的,、()將一組表達式串起來執行。也就是說,這個表達式可能會很長很長,我甚至可以定義一個function在裡面。這些表達式在執行過程中,有可以進行某些附加操作。比如我們希望這個表達式為真值的時候我們做什麼,假值的時候做什麼,把這些操作用()、,串起來作為一個整體運算。

於是就有瞭上面的復雜代碼。

另外:

if(a){
    b
}

可簡寫為a&&(b); b可以是一個function調用表達式,或者是多個語句用","串起來。但前提是a已定義,否則會報錯。

我們來看個實例吧。是上面例子的升級版。我們加入一個nameInput是否存在的判斷:

jQuery(document).ready(function() {
    var nameValidate=function(){
        var value,nameInput=$("#name"),nameErrorTip=$("#nameErrorTip"),msg;
        msg=(value=nameInput.val(),value=="king")?(nameErrorTip.hide(),"對瞭,輸入為king!"):(nameErrorTip.show(),"請輸入king!");
        return (nameInput.length&&nameInput.val()&&nameErrorTip.length&&msg)||"沒有找到name輸入框或者輸入框沒有值!";
    };
    alert(nameValidate());
});

測試:

//html:
//結果:彈出“沒有找到name輸入框或者輸入框沒有值!”

//

輸入錯誤!

//結果:彈出“對瞭,輸入為king!”,nameErrorTip被隱藏

return表示中nameInput.length&&nameInput.val()&&nameErrorTip.length&&msg會先運算nameInput.length的值,如果length為0則表達式為假值,如果為1則為真值。val()操作也是如此,如果val()結果為""則表達式也是假值。幾個表達式之間為&&運算,則表示依次運算幾個表達式的值,如果都未真值則返回最後一個表達式的值,由於整個表達式與"沒有找到name輸入框或者輸入框沒有值!"表達式之間是||運算,所以前面的表達式其中一個表達式為假值則返回||右側的表達式的值,也就是整個“沒有找到name輸入框或者輸入框沒有值!”字符串。

說到這裡,我之前寫過一篇文章專門說到瞭&&、||的真值、假值問題。有興趣的可以去看看。

談瞭這些難以理解的運算符後,大傢可能會覺得,這個javascript為什麼要搞這些晦澀的運算符呢?

我的理解是因為javascript通常在客戶端運行,那麼從服務器端將js代碼傳輸到客戶端肯定需要耗時。上面的這些運算符都是為瞭減少代碼量。再加上使用壓縮工具去掉空格,替換變量名,就可以使用壓縮率達到最好。

再這裡,我也告訴大傢,其實我也非常反對在實際應用中采用這種寫法的,因為會對初學者造成閱讀障礙。我寫這篇文章的目的不是為瞭讓大傢以後就這樣用,而是告訴大傢可以這樣用,在一些開源代碼中遇到瞭能看懂。不可為瞭炫耀而故意寫一些晦澀的代碼。

最後,為瞭幫助我們更快的找到變量定義、理清代碼整體結構,給大傢推薦一個eclipse的js插件:Spket,支持jQuery代碼提示哦!

發佈留言