本文實例講述了jQuery源碼解讀之extend()與工具方法、實例方法。分享給大家供大家參考,具體如下:
使用jQuery的時候會發現,jQuery中有的函數是這樣使用的:
$.get(); $.post(); $.getJSON();
有些函數是這樣使用的:
$('div').css(); $('ul').find('li');
有些函數是這樣使用的:
$('li').each(callback); $.each(lis,callback);
這裡涉及到兩個概念:工具方法與實例方法。通常我們說的工具方法是指無需實例化就可以調用的函數,如第一段代碼;實例方法是必須實例化對象以後才可以調用的函數,如第二段代碼。jQuery中很多方法既是實例方法也是工具方法,只是調用方式略有不同,如第三段代碼。為了更清晰解釋Javascript中的工具方法與實例方法,進行如下測試。
function A(){ } A.prototype.fun_p=function(){console.log("prototpye");}; A.fun_c=function(){console.log("constructor");}; var a=new A(); A.fun_p();//A.fun_p is not a function A.fun_c();//constructor a.fun_p();//prototpye a.fun_c();//a.fun_c is not a function
通過以上測試可以得出結論,在原型中定義的是實例方法,在構造函數中直接添加的是工具方法;實例方法不能由構造函數調用,同理,工具方法也不能由實例調用。
當然實例方法不僅可以在原型中定義,有以下三種定義方法:
function A(){ this.fun_f=function(){ console.log("Iam in the constructor"); }; } A.prototype.fun_p=function(){ console.log("Iam in the prototype"); }; var a=new A(); a.fun_f();//Iam in the constructor a.fun_i=function(){ console.log("Iam in the instance"); }; a.fun_i();//Iam in the instance a.fun_p();//Iam in the prototype
這三種方式的優先級為:直接定義在實例上的變量的優先級要高於定義在“this”上的,而定義在“this”上的又高於 prototype定義的變量。即直接定義在實例上的變量會覆蓋定義在“this”上和prototype定義的變量,定義在“this”上的會覆蓋prototype定義的變量。
下面看jQuery中extend()方法源碼:
jQuery.extend = jQuery.fn.extend = function() { var options,name, src, copy, copyIsArray, clone, target= arguments[0] || {}, i =1, length= arguments.length, deep= false; // Handle adeep copy situation if ( typeoftarget === "boolean" ) { deep= target; //Skip the boolean and the target target= arguments[ i ] || {}; i++; } // Handlecase when target is a string or something (possible in deep copy) if ( typeoftarget !== "object" && !jQuery.isFunction(target) ) { target= {}; } // ExtendjQuery itself if only one argument is passed if ( i ===length ) { target= this; i--; } for ( ; i< length; i++ ) { //Only deal with non-null/undefined values if ((options = arguments[ i ]) != null ) { //Extend the base object for( name in options ) { src= target[ name ]; copy= options[ name ]; //Prevent never-ending loop if( target === copy ) { continue; } //Recurse if we're merging plain objects or arrays if( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray= jQuery.isArray(copy)) ) ) { if( copyIsArray ) { copyIsArray= false; clone= src && jQuery.isArray(src) ? src : []; }else { clone= src && jQuery.isPlainObject(src) ? src : {}; } //Never move original objects, clone them target[name ] = jQuery.extend( deep, clone, copy ); //Don't bring in undefined values }else if ( copy !== undefined ) { target[name ] = copy; } } } } // Returnthe modified object return target; };
(1)首先,jQuery和其原型中extend()方法的實現使用的同一個函數。
(2)當extend()中只有一個參數的時候,是為jQuery對象添加插件。在jQuery上擴展的叫做工具方法,在jQuery.fn(jQuery原型)中擴展的是實例方法,即使在jQuery和原型上擴展相同名字的函數也可以,使用jQuery對象會調用工具方法,使用jQuery()會調用實例方法。
(3)當extend()中有多個參數時,後面的參數都擴展到第一個參數上。
var a={}; $.extend(a,{name:"hello"},{age:10}); console.log(a);//Object{name: "hello", age: 10}
(4)淺拷貝(默認):
var a={}; varb={name:"hello"}; $.extend(a,b); console.log(a);//Object{name: "hello"} a.name="hi"; console.log(b);//Object{name: "hello"}
b不受a影響,但是如果b中一個屬性為對象:
var a={}; varb={name:{age:10}}; $.extend(a,b); console.log(a.name);//Object{age: 10} a.name.age=20; console.log(b.name);//Object{age: 20}
由於淺拷貝無法完成,則b.name會受到a的影響,這時我們往往希望深拷貝。
深拷貝:
var a={}; varb={name:{age:10}}; $.extend(true,a,b); console.log(a.name);//Object{age: 10} a.name.age=20; console.log(b.name);//Object{age: 10}
b.name不受a的影響。
var a={name:{job:"Web Develop"}}; var b={name:{age:10}}; $.extend(true,a,b); console.log(a.name);//age: 10 job: "Web Develop" //b.name沒有覆蓋a.name.job。
更多關於jQuery相關內容感興趣的讀者可查看本站專題:《jQuery擴展技巧總結》、《jQuery常用插件及用法總結》、《jQuery切換特效與技巧總結》、《jQuery遍歷算法與技巧總結》、《jQuery常見經典特效匯總》、《jQuery動畫與特效用法總結》及《jquery選擇器用法總結》
希望本文所述對大家jQuery程序設計有所幫助。