zoukankan      html  css  js  c++  java
  • jQuery对象扩展方法(Extend)深度解析

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

    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;
    };

    下面是我的解释代码:

    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)、浅拷贝代码:

    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)、深拷贝代码:

    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);

  • 相关阅读:
    Leetcode 58. 最后一个单词的长度 双指针
    Leetcode 125. 验证回文串 双指针
    拜托,大厂做项目可不简单!
    被问懵了:一个进程最多可以创建多少个线程?
    面对祖传屎山代码应该采用的5个正确姿势
    VUE代码格式化配置vetur、eslint、prettier的故事
    如何快速实现一个虚拟 DOM 系统
    NodeJS 进程是如何退出的
    [堆][启发式合并]luogu P3261 [JLOI2015]城池攻占
    [Trie][堆]luogu P5283 [十二省联考2019]异或粽子
  • 原文地址:https://www.cnblogs.com/GreenLeaves/p/6890030.html
Copyright © 2011-2022 走看看