深拷贝与浅拷贝的实现(一)

最近的学习中,仔细研究了下深拷贝和浅拷贝,下面就来简单的总结下。

数据类型

首先我们了解下两种数据类型

1、基本类型:像Number、String、Boolean等这种为基本类型

2、复杂类型:Object和Array

浅拷贝与深拷贝的概念

接着我们分别来了解下浅拷贝和深拷贝,深拷贝和浅拷贝是只针对Object和Array这样的复杂类型的。

浅拷贝

var a = {

    myname: 'yana'

};

var b = a;

b.myname = '小雅';

console.log(b.myname);    // 小雅

console.log(a.myname);    // 小雅



var a = ['myname', 'yana'];

var b = a;

b[1] = '小雅';

console.log(a);    // ["myname", "小雅"]

console.log(b);    // ["myname", "小雅"]


可以看出,对于对象或数组类型,当我们将a赋值给b,然后更改b中的属性,a也会随着变化。

也就是说a和b指向了同一块内存,所以修改其中任意的值,另一个值都会随之变化,这就是浅拷贝。

深拷贝

刚刚我们了解了什么是浅拷贝,那么相应的,如果给b放到新的内存中,将a的各个属性都复制到新内存里,就是深拷贝。

也就是说,当b中的属性有变化的时候,a内的属性不会发生变化。

浅拷贝

那么除了上面简单的赋值引用,还有哪些方法使用了浅拷贝呢?

Object.assign()

在MDN上介绍Object.assign():”Object.assign() 方法用于将所有可枚举的属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。”

复制一个对象

var target = {a: 1, b: 1};

var copy1 = {a: 2, b: 2, c: {ca: 21, cb: 22, cc: 23}};

var copy2 = {c: {ca: 31, cb: 32, cd: 34}};

var result = Object.assign(target, copy1, copy2);

console.log(target);    // {a: 2, b: 2, c: {ca: 31, cb: 32, cc: 33}}

console.log(target === result);    // true


可以看到,Object.assign()拷贝的只是属性值,假如源对象的属性值是一个指向对象的引用,它也只拷贝那个引用值。所以Object.assign()只能用于浅拷贝或是合并对象。这是Object.assign()值得注意的地方。

深拷贝

那么下面我们就来说说复杂的深拷贝

jQuery.extend()

说到深拷贝,第一想到的就是jQuery.extend()方法,下面我们简单看下jQuery.extend()的使用。

jQuery.extend( [deep ], target, object1 [, objectN ] ),其中deep为Boolean类型,如果是true,则进行深拷贝。

我们还是用上面的数据来看下extend()方法。

var target = {a: 1, b: 1};

var copy1 = {a: 2, b: 2, c: {ca: 21, cb: 22, cc: 23}};

var copy2 = {c: {ca: 31, cb: 32, cd: 34}};

var result = $.extend(true, target, copy1, copy2);   // 进行深拷贝

console.log(target);    // {a: 2, b: 2, c: {ca: 31, cb: 32, cc: 23, cd: 34}}


var target = {a: 1, b: 1};

var copy1 = {a: 2, b: 2, c: {ca: 21, cb: 22, cc: 23}};

var copy2 = {c: {ca: 31, cb: 32, cd: 34}};

var result = $.extend(target, copy1, copy2);   // 不进行深拷贝

console.log(target);    // {a: 1, b: 1, c: {ca: 31, cb: 32, cd:34}}


通过上面的对比可以看出,当使用extend()进行深拷贝的时候,对象的所有属性都添加到target中了。

我们知道了extend()可以进行深拷贝,那么extend()是如何实现深拷贝的呢?

先来看下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 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;

};

主要看下关于深拷贝的部分,取第一个参数,如果是boolean类型的,就赋值给deep,下面如果deep为true(也就是进行深拷贝),就递归调用extend(),这样就将对象的所有属性都添加到了target中实现了深拷贝。

JSON.parse()和JSON.stringify()

上面的jQuery源码是否让你眼花缭乱?有没有什么办法无脑实现深拷贝呢?JSON.parse()和JSON.stringify()给了我们一个基本的解决办法。

var target = {a: 1, b: 1, c: {ca: 11, cb: 12, cc: 13}};

var targetCopy = JSON.parse(JSON.stringify(target));

targetCopy.a = 2;

targetCopy.c.ca = 21;

console.log(target);   // {a: 1, b: 1, c: {ca: 11, cb: 12, cc: 13}}

console.log(targetCopy);    // {a: 2, b: 1, c: {ca: 21, cb: 12, cc: 13}}

console.log(target === targetCopy);  // false


可以看到改变targetCopy并没有改变原始的target,继承的属性也没有丢失,因此实现了基本的深拷贝。

但是用JSON.parse()和JSON.stringify()会有一个问题。

JSON.parse()和JSON.stringify()能正确处理的对象只有Number、String、Array等能够被json表示的数据结构,因此函数这种不能被json表示的类型将不能被正确处理。

var target = {

    a: 1,

    b: 2,

    hello: function() {

            console.log("Hello, world!");

    }

};

var copy = JSON.parse(JSON.stringify(target));

console.log(copy);   // {a: 1, b: 2}


上面的例子可以看出,hello这个属性由于是函数类型,使用JSON.parse()和JSON.stringify()后丢失了。

因此JSON.parse()和JSON.stringify()还是需要谨慎使用。

下篇文章我会继续为大家说明深拷贝的各种实现。

未完待续……

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容