前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >jQuery对象扩展方法(Extend)深度解析

jQuery对象扩展方法(Extend)深度解析

作者头像
郑小超.
发布2018-01-26 15:12:40
7720
发布2018-01-26 15:12:40
举报
文章被收录于专栏:GreenLeavesGreenLeaves

1、这几天在写自己的Js工具类库,所以在编写对象扩展方法,参考了jQuery的对象扩展方法,在编写该方法前,需要掌握js深拷贝和浅拷贝的相关知识,下面是jQuery3.2.1版本对象扩展方法的源码:

代码语言:javascript
复制
jQuery.extend = jQuery.fn.extend = function() {
    var options, name, src, copy, copyIsArray, clone,
        target = arguments[ 0 ] || {},
        i = 1,
        length = arguments.length,
        deep = false;

    // Handle a deep copy situation
    if ( typeof target === "boolean" ) {
        deep = target;

        // Skip the boolean and the target
        target = arguments[ i ] || {};
        i++;
    }

    // Handle case when target is a string or something (possible in deep copy)
    if ( typeof target !== "object" && !jQuery.isFunction( target ) ) {
        target = {};
    }

    // Extend jQuery 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 = Array.isArray( copy ) ) ) ) {

                    if ( copyIsArray ) {
                        copyIsArray = false;
                        clone = src && Array.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;
                }
            }
        }
    }

    // Return the modified object
    return target;
};

下面是我的解释代码:

代码语言:javascript
复制
zcHtmlHelper.extend=zcHtmlHelper.fn.extend=function(){
    var target=arguments[0]||{},//第一个参数
        deep = false,//是否开启深拷贝功能,默认不是
        i=1,
        length=arguments.length,
        options,
        name,
        src,
        copy,
        copyIsArray,
        clone;

    //处理深拷贝场景
    if(typeof target==="boolean"){
        deep=true;
        target=arguments[i] || {};//将紧随其后的存放拷贝值的集合
        i++;//加1的原因是,一旦开启深拷贝功能,那么传入的参数就至少有两个,一个是深拷贝的开关另一个是扩展参数,否则当只传如一个深拷贝的开关,那么方法将返回空集合
    }

    if(typeof target!=="object"){
        target = {};
    }

    //这个判断有两种情况
    //1、当传入的参数只有一个(不能是true或者false),那么就扩展当前命名空间
    //2、当传入的参数有个两个,分别是深拷贝的开关(true或者false)和扩展参数,那么就扩展当前命名空间
    if(i==length){
        target=this;
        i--;
    }
    for(;i<length;i++){
        //只处理非空和非null的值
        if((options=arguments[i])!=null){
            for(name in options){
                src=target[name];//判断传入的参数(存放拷贝值的集合)
                copy=options[name];

                //当嵌套数组或者对象深拷贝完毕,跳出当前属性,开始拷贝下一属性
                if ( target === copy ) {
                    continue;
                }

                //如果传入的合并对象里面嵌套数组或者对象,那么递归扩展对象
                if(deep && copy && (zcHtmlHelper.isPlainObject(copy) || (copyIsArray = Array.isArray(copy))))
                {
                    if ( copyIsArray ) {
                        copyIsArray = false;
                        clone = src && Array.isArray( src ) ? src : [];
                    }
                    else {
                        clone = src && zcHtmlHelper.isPlainObject( src ) ? src : {};
                    }
                    target[ name ] = zcHtmlHelper.extend( deep, clone, copy );
                }
                else if(copy!=undefined)
                {
                    target[name]=copy;//覆盖添加
                }
            }
        }
    }
    return target;
}

2、代码验证

(1)、浅拷贝代码:

代码语言:javascript
复制
var names=[1,3,4,5,6];
var defaults = { validate: false, limit: 5, name: "foo" };
var options = { names: names};
var settings = zcHtmlHelper.extend(false,defaults,options);
alert(names==settings.names);

首先对象拷贝成功,settings是两个对象的合集,但是name数组对象和settings.name属性是同一个引用,所以,这是前拷贝

(2)、深拷贝代码:

代码语言:javascript
复制
var names=[1,3,4,5,6];
var defaults = { validate: false, limit: 5, name: "foo" };
var options = { names: names};
var settings = zcHtmlHelper.extend(true,defaults,options);
alert(names==settings.names);
console.log(settings);
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2017-05-22 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档