关于Javascript对象拷贝实现及疑问实例

在实际编程过程,有时候我们会遇到一种情况:当你有一个对象A,在某一个时刻,A已经保存了对应的属性值,而且这些值本身是有效的,这个时候可能需要一个和A完全相同的对象B,并且当B里面的属性值发生变化的时候,A中的属性值不受影响,可以理解为A和B独立,但是B的初始化不是按照我们平时创建该对象的时候的初始化操作,B的初始化数据完全来自A。

一、浅拷贝

当我们需要将一个对象拷贝至另一个对象时,我们一般会这么实现

function shadowCopy(source,target){
    var target=target||{};
    for(var i in source)
    {
        target[i]=source[i];
    }
    return target;
}
var a={name:'Lily',age:19};
var b=shadowCopy(a);//b={name:'Lily',age:19}

浅拷贝的问题是,如果父对象的属性等于数组或另一个对象,实际上子对象获得的只是一个内存地址,而不是真正拷贝,父对象的数组或对象属性发生变化时,子对象对应属性也发生变化

function shadowCopy(source,target){
    var target=target||{};
    for(var i in source)
    {
        target[i]=source[i];
    }
    return target;
}
var a={name:'Lily',Hobbies:['Music','Sport']};
var b=shadowCopy(a);//b={name:'Lily',Hobbies:['Music','Sport']}
a.Hobbies.push('Read');//b={name:'Lily',Hobbies:['Music','Sport','Read']}

二、深拷贝

为了解决上述问题,需要对对象的数组和对象属性进行深拷贝。它的实现并不难,只要递归调用"浅拷贝"就行了

function deepCopy(source,target){
    var target=target||{};
    for(var i in source)
    {
        if(typeof source[i] === 'object'){
            target[i] = (source[i].constructor === Array ) ? [] : {} ;
            deepCopy(source[i],target[i]);
        }else{
            target[i]=source[i];
        }
    }
    return target;
}             
var a={name:'Lily',Hobbies:['Music','Sport']};
var b=deepCopy(a);//b={name:'Lily',Hobbies:['Music','Sport']}
a.Hobbies.push('Read');//b={name:'Lily',Hobbies:['Music','Sport','Read']},b={name:'Lily',Hobbies:['Music','Sport']}

上述代码中有一个问题,当待拷贝对象中存在自引用时,程序会陷入无限循环

var a={name:'lily'};
a.obj=a;
deepCopy(a);

在Chome Console运行时,如下提示

RangeError: Maximum call stack size exceeded

为了解决自引用问题,拷贝时加入判断逻辑

function deepCopy(source,target){
    var target=target||{};
    for(var i in source)
    {
                //防止自引用
        if(source[i] === source )
            continue;
        if(typeof source[i] === 'object'){
            target[i] = (source[i].constructor === Array ) ? [] : {} ;
            deepCopy(source[i],target[i]);
        }else{
            target[i]=source[i];
        }
    }
    return target;
}             
var a={name:'lily'};
a.obj=a;
var b=deepCopy(a);//b={name:'lily'}

三、JQuery拷贝实现

网上有很多对JQuery extend方法的分析,有不了解的可以去搜索阅读

贴一处被分析的源码

jQuery.extend = jQuery.fn.extend = function() {
    var src, copyIsArray, copy, name, options, clone,
        target = arguments[0] || {},
        i = 1,
        length = arguments.length,
        deep = false;
    // Handle a deep copy situation
    if ( typeof target === "boolean" ) {
        deep = target;
        target = arguments[1] || {};
        // skip the boolean and the target
        i = 2;
    }
    // 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 ( length === i ) {
        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;
                }
            }
        }
    }
    // Return the modified object
    return target;
};

四、JQuery实现疑问

在阅读上述JQuery代码时,有个地方有疑问,疑问代码如下

// Prevent never-ending loop
if ( target === copy ) {
    continue;
}

注意到,在extend方法中,为了防止无限循环,这里有一个逻辑,在target对象等于copy对象时,调过这次复制操作。其中,copy对象为options对象的属性对象。

问题是,这里为什么要拿target对象与copy对象比较呢?难道不应该是比较copy对象和options对象吗?

带着这个疑问,在一个已经引入了JQuery库的页面Console中执行下

var a={name:'lily'};
a.obj=a;
var b={};
$.extend(true,b,a);
RangeError: Maximum call stack size exceeded

可以看到,当a对象中存在自引用属性时,extend方法并不能防止无限循环的发生

那么判断target === copy能起到什么作用呢?

var a={name:'lily'};
var b={age:19};
a.obj=b;
$.extend(true,b,a);
//此时b={age: 19, name: "lily"}

去掉判断target === copy会陷入无限循环吗?实际上是不会的

var a={name:'lili'};
var b={age:19};
a.obj=b;
deepCopy(true,b,a);
//b=Object {age: 19, name: "lili", obj: Object}
//其中Object为b

这里的deepCopy是我将JQuery的extend方法,去掉上述判断逻辑,自己实现了一份

function deepCopy() {
    var src, copyIsArray, copy, name, options, clone,
        target = arguments[0] || {},
        i = 1,
        length = arguments.length,
        deep = false;
    // Handle a deep copy situation
    if ( typeof target === "boolean" ) {
        deep = target;
        target = arguments[1] || {};
        // skip the boolean and the target
        i = 2;
    }
    // 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 ( length === i ) {
        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 ( options === copy ) {
                    continue;
                }
                */
                // Recurse if we're merging plain objects or arrays
                if ( deep && copy && ( isPlainObject(copy) || (copyIsArray =isArray(copy)) ) ) {
                    if ( copyIsArray ) {
                        copyIsArray = false;
                        clone = src && isArray(src) ? src : [];
                    } else {
                        clone = src && isPlainObject(src) ? src : {};
                    }
                    // Never move original objects, clone them
                    target[ name ] =deepCopy( deep, clone, copy );
                // Don't bring in undefined values
                } else if ( copy !== undefined ) {
                    target[ name ] = copy;
                }
            }
        }
    }
    // Return the modified object
    return target;
};
var isString=function(obj){
    return Object.prototype.toString.call(obj) === '[object String]';
};
var isArray=function(obj){
    return Object.prototype.toString.call(obj) === '[object Array]';
};
var isPlainObject=function(obj){
    return Object.prototype.toString.call(obj) === '[object Object]';
}
var a={name:'lili'};
var b={age:19};
a.obj=b;
deepCopy(true,b,a);

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索数组
, 递归
, 对象
属性
javascript 对象拷贝、javascript实例化对象、javascript对象实例、javascript 深拷贝、javascript 深度拷贝,以便于您获取更多的相关知识。

时间: 2024-07-30 16:50:58

关于Javascript对象拷贝实现及疑问实例的相关文章

如何判断Javascript对象是否存在的简单实例_javascript技巧

Javascript语言的设计不够严谨,很多地方一不小心就会出错. 举例来说,请考虑以下情况. 现在,我们要判断一个全局对象myObj是否存在,如果不存在,就对它进行声明.用自然语言描述的算法如下: if (myObj不存在){ 声明myObj; } 你可能会觉得,写出这段代码很容易.但是实际上,它涉及的语法问题,远比我们想象的复杂.Juriy Zaytsev指出,判断一个Javascript对象是否存在,有超过50种写法.只有对Javascript语言的实现细节非常清楚,才可能分得清它们的区别

JavaScript对象创建及继承原理实例解剖_基础知识

对象创建: 当一个函数对象被创建时候,Function构造器产生的函数对象会运行类似这样的代码: 复制代码 代码如下: this.prototype={constructor:this}; 假设函数F F用new方式构造对象时,对象的constructor被设置成这个F.prototype.constructor 如果函数在创建对象前修改了函数的prototype,会影响创建出来对象的construtor属性 如: 复制代码 代码如下: function F(){}; F.prototype={

Javascript对象Clone实例分析

 本文实例讲述了Javascript对象Clone用法.分享给大家供大家参考.具体如下:   1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 Object.prototype.Clone=function() { var objClone=new this.constructor(); //这里是创建一个与被Clone对象相同结构的对象 for(var key in this) { if(obj

微信浏览器内置JavaScript对象WeixinJSBridge使用实例

  这篇文章主要介绍了微信浏览器内置JavaScript对象WeixinJSBridge使用实例,本文给出了分享到朋友圈.发送给好友.分享到腾讯微博.关注指定的微信号等功能代码,需要的朋友可以参考下 微信公众平台开始支持前端网页,大家可能看到很多网页上都有分享到朋友圈,关注微信等按钮,点击它们都会弹出一个窗口让你分享和关注,这个是怎么实现的呢?今天就给大家讲解下如何在微信公众平台前端网页上添加分享到朋友圈,关注微信号等按钮. 一.微信内置浏览器 通过 Mac 远程调试 iPhone 上微信自己的

对象-javascript中经常提到的【实例】是什么意思?

问题描述 javascript中经常提到的[实例]是什么意思? 经常在博客或者教程里看到 实例化,或者实例这个词 比如创建一个对象实例 这样. 但是一直不是很清楚这个实例具体是什么意思, 求解! 解决方案 就是类的实例,如果学过面向对象编程就很容易明白了 function MyClass(){} var m=new MyClass();//m就是MyClass的实例,MyClass是类,new就是创建操作符 解决方案二: 对象就是类的实例化,类是对某一类事物的描述,并不实实在在存在,而new一个

JavaScript对象属性检查、增加、删除、访问操作实例_javascript技巧

检查属性 var mouse = { "name": "betta", "age": 3, "varieties": "milaoshu" } mouse.hasOwnProperty("name"); // true mouse.hasOwnProperty("sex"); //false 增加属性 定义个对象 dog,然后赋予各种特性,再赋予 color特性,最后

javascript 对象入门实例教程_js面向对象

1:构造函数法 [Ctrl+A 全选 注:如需引入外部Js需刷新才能执行] 注解: 1:这里定义了一个Dog对象(在javascript中,函数就是对象,这里function Dog(name,weight) 同时也是构造函数),用new关键字创建了一个对象实例dog. 2:其中_name._weight._show._sex为实例dog的属性.可以通过:实例名.属性名 或 实例名["属性名"]访问实例属性,即dog._name=dog["_name"]. 3:在构

JavaScript对象创建模式实例汇总_javascript技巧

本文实例总结了JavaScript对象创建模式.分享给大家供大家参考,具体如下: 在JavaScript中创建对象是很容易的,可以使用对象字面量或者构造函数.常用的创建对象的模式有以下几种: 一. 工厂模式 工厂模式抽象了具体对象的过程,用函数来封装以特ing接口创建对象的细节. 如下: function createAnimal(name, age) { var o = new Object(); o.name = name; o.age = age; o.sayName = function

JavaScript对象反射用法实例_javascript技巧

本文实例讲述了JavaScript对象反射用法.分享给大家供大家参考.具体如下: 这里讲述JavaScript对象反射用法,涉及反射DOM对象和自定义对象 <html> <head> <title>JavaScript反射工具</title> <style type="text/css"> #show{ width:400px;height:300px; border:red solid 1px; overflow:scrol