码迷,mamicode.com
首页 > Web开发 > 详细

JQuery 1.10.2源码分析学习4(转载)

时间:2015-06-09 16:31:34      阅读:159      评论:0      收藏:0      [点我收藏+]

标签:

//这里是插件接口
jQuery.extend = jQuery.fn.extend = function() {
    var src, copyIsArray, copy, name, options, clone,
        target = arguments[0] || {}, //参数数组arguments
        i = 1,
        length = arguments.length,
        deep = false;
    if ( typeof target === "boolean" ) {
        deep = target;
        target = arguments[1] || {};
        i = 2;
    }
    if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
        target = {};
    }
    if ( length === i ) {
        target = this;
        --i;
    }
    for ( ; i < length; i++ ) {
        if ( (options = arguments[ i ]) != null ) {
            for ( name in options ) {
                src = target[ name ];
                copy = options[ name ];
                if ( target === copy ) {
                    continue;
                }
                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 : {};
                    }
                    target[ name ] = jQuery.extend( deep, clone, copy );
                } else if ( copy !== undefined ) {
                    target[ name ] = copy;
                }
            }
        }
    }
    return target;
};

 

jQuery为开发插件提供了两个方法,分别是:
jQuery.fn.extend(Object); 给jQuery对象添加方法。

jQuery.extend(Object); 为扩展jQuery类本身.为类添加新的方法。

 

fn 是什么东西呢。查看jQuery代码,就不难发现。

jQuery.fn = jQuery.prototype = {

   init: function( selector, context ) {//.... 

   //......

};

原来 jQuery.fn = jQuery.prototype.对prototype肯定不会陌生啦。

虽然 javascript 没有明确的类的概念,但是用类来理解它,会更方便。

jQuery便是一个封装得非常好的类,比如我们用 语句 $("#btn1") 会生成一个 jQuery类的实例。

jQuery.extend(object); 为jQuery类添加添加类方法,可以理解为添加静态方法。如:

$.extend({

  add:function(a,b){return a+b;}

});

便为 jQuery 添加一个为 add 的 “静态方法”,之后便可以在引入 jQuery 的地方,使用这个方法了,

$.add(3,4);  //return 7(这里是类名.方法名)

 

jQuery.fn.extend(object); 对jQuery.prototype进得扩展,就是为jQuery类添加“成员函数”。jQuery类的实例可以使用这个“成员函数”。

比如我们要开发一个插件,做一个特殊的编辑框,当它被点击时,便alert 当前编辑框里的内容。可以这么做:

  1. $.fn.extend({    
  2.     
  3.    alertWhileClick:function(){    
  4.    
  5.        $(this).click(function(){    
  6.    
  7.             alert($(this).val());    
  8.         });    
  9.     
  10.     }    
  11.     
  12. });

$("#input1").alertWhileClick(); //页面上为:<input id="input1" type="text"/>

(这里是对象.方法名

 

真实的开发过程中,当然不会做这么小白的插件,事实上jQuery提拱了丰富的操作文档,事件,CSS ,Ajax、效果的方法,结合这些方法,便可以开发出更加 Niubility 的插件。(转)

jQuery.extend()使用方式有三种: 
1.jQuery.extend({...}),参数为一个对象,这种使用方式是把参数的值cp给jQuery对象,这是扩展jQuery最常用的方式,后面很多代码都采用这种方式。 
2.jQuery.extend({...},{...}...),参数为多个对象,这种方式是把后面的多个对象的值cp给第一个对象,如果后面对象有和前面对象相同的属性,则后面的覆盖前面的。 
3.jQuery.extend(true,{a:{h:22,w:333}},{a:{h:552,s:2532},c:287}),深度cp,递归合并、覆盖、复制的方式。返回结果为{a:{h:552,w:333,s:2532},c:287}

-----------------------------------------------------------

我们可以用$.extend去扩展自定义的对象,如


var myself = {name:jack}; 
$.extend(myself, {setName: function(n){this.name=n;} }); 
myself.setName("tom");

-----------------------------------------------------------------

通过$.extend为对象myself添加了setName方法。但这里主要讨论$.extend如何构建jQuery库的。不知注意到上面代码中jQuery.extend和jQuery.fn.extend是同一个函数。

我们浏览下jQuery库,发现有些方法是通过调用jQuery.extend扩展的,有些则是通过调用jQuery.fn.extend扩展的。 


下面分别讨论:

 

1,通过jQuery.extend扩展 
我们知道jQuery.extend中的jQuery类型是function,即typeof jQuery值为字符串“function”。

如果把jQuery当成一个类,jQuery.extend相当于为该类添加了静态方法extend。

静态方法是不需要new一个实例再去调用的,通过“类名+方法名”直接调用。

即jQuery.extend(...),jQuery又被赋值给$。因此我们更习惯$.extend(...)。 
源码中直接调用jQuery.extend方法,只传一个参数。如下

jQuery.extend({ 
  noConflict: function( deep ) { 
    window.$ = _$; 
    if ( deep ) 
      window.jQuery = _jQuery; 
    return jQuery; 
    }, 
    ... 
});

我们知道extend中如果只传一个参数,那么将执行该句
if ( length === i ) {

  target = this; 
  --i; 
}

即扩展自己,而这里的this则是function jQuery。

也就是说给function jQuery添加了许多静态方法,

这些方法都可以直接通过jQuery.xx(或$.xx)方式来调用,

而不是先执行(调用)jQuery方法再去调用xx,如$("#id").xx。

也许下面这个例子更容易理解

function fun(){}//定义一个类(函数) 
//为该类(函数)添加一个静态方法extend 
fun.extend=function(obj){ 
for(var a in obj) 
this[a] = obj[a];//注意:这里的tihs即fun 

//调用extend为该类添加了静态属性name,静态方法method1 
fun.extend({name:"fun",method1:function(){}}); 
//输出name,prototype,extend,method1 
console.dir(fun)

因此,jQuery中的isFunction, isArray, isWindow等都是静态方法,

只能通过$.isFunction, $.isArray, $.Window引用。

而不能通过$("...").isFuction, $("...").isArray, $("...").isWindow方式引用。

 

2,通过jQuery.fn.extend扩展 
jQuery.fn等于jQuery.prototype,也就是说给function jQuery的原型(prototype)上挂了个extend方法。通过调用jQuery.fn.extend(object)来扩展时(注意只传一个参数object),extend函数中仍然会执行

if ( length === i ) { 
target = this; 
--i; 
}

而这时的this则是jQuery.prototype(第一条提到的则是jQuery函数自身)。

即给jQuery.prototype上添加了许多属性,方法。

当jQuery函数执行时,如$()或jQuery(),更多时候是$()。

该函数会new一个jQuery(见上一篇jQuery对象的组成)。

这时则把扩展的属性,方法都附加到new生成的对象上了。

也许下面这个示例更容易理解


function fun(){}//定义一个类(函数) 
//给该类原型上添加一个方法extned 
fun.prototype.extend = function(obj){ 
for(var a in obj) 
this[a] = obj[a];//注意:这里的this即是fun.prototype 

//调用extend方法给fun.prototype上添加属性,方法 
fun.prototype.extend({name:"fun2",method1:function(){}}) 
//输出name,extend,method1 
console.dir(new fun())

因此扩展的属性或方法都添加到jQuery对象上了。如bind, one, unbind等可以通过$("...").bind, $("...").one, $("...").unbind方式调用。却不能通过 $.bind, $.one, $.unbind方式调用。 
jQuery与Prototype一样都是通过extend方法扩展出整个库的。相对来说jQuery的扩展方式更难理解一些。 


总结如下: 
1,jQuery.extend({...})是给function jQuery添加静态属性或方法。 
2,jQuery().extend({...})是给jQuery对象添加属性或方法。

 

今天所说的基本都是转来的文章。我觉得看懂这些应该对JQuery实现扩展有了更深层次的理解吧!!

 

 

原创:http://blog.csdn.net/zzx252373003/article/details/11106693

JQuery 1.10.2源码分析学习4(转载)

标签:

原文地址:http://www.cnblogs.com/mjzhang/p/4563303.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!