Javascript中的浅拷贝和深拷贝

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Javascript中的浅拷贝和深拷贝相关的知识,希望对你有一定的参考价值。

很多开发语言中都有浅拷贝和深拷贝的说法,这里简单区分一下它们在javascript中的区别,以及jQuery中深拷贝的实现。

在谈浅拷贝和深拷贝之前,先要屡清楚Javascript中的按值访问和按引用访问这两个概念。

按值访问是针对基本类型(string、number、boolean、null、undefined)。基本类型以值的形式被存放在栈内存中,我们通过变量操作的是栈内存中实际的值。

按引用访问时针对引用类型(Object、Array、Date、RegExp、Function)。引用类型以对象的形式被存放在堆内存中,我们通过变量操作的堆内存中对象的引用,也就是说,变量中保存的是一个指针,指向内存中的某个位置。

--------------------------分割线--------------------------

浅拷贝与深拷贝的区别是针对引用类型而言的,浅拷贝只是拷贝了对象表面上的一层,而深拷贝是拷贝了对象的所有层级。

那这两种方式,区别在哪里呢?

举个例子:

var arr = [‘a‘, ‘b‘];
var newArr = arr;  // newArr拷贝了arr

console.log(newArr);  // [‘a‘, ‘b‘]

arr.push(‘c‘);

console.log(newArr);  //[‘a‘, ‘b‘, ‘c‘]

以上代码是一个简单的浅拷贝。

由于arr是一个数组,数组内部保存了两个string类型的值。newArr通过直接赋值得到该数组的引用,此时newArr和arr所指向的是内存中的同一个位置,所以arr的操作反应到内存中数组的改变,也就导致了newArr的改变。

下面来看看深拷贝:

var arr = [‘a‘, ‘b‘];
var newArr = [];

newArr[0] = arr[0];
newArr[1] = arr[1];

console.log(newArr);  // [‘a‘, ‘b‘]

arr.push(‘c‘);

console.log(newArr);  //[‘a‘, ‘b‘]

以上代码是一个简单的深拷贝。

我们发现此时,arr的改变并没有影响到newArr,原因就在于,这次我们是先定义了一个空数组newArr(在内存中开辟了新的位置),此时newArr与arr所指向的是内存中不同的位置,然后我们对数组中的string类型的值进行赋值操作,得到了最后的newArr。

那么,我们现实中遇到的拷贝问题不会是这简单的数组,可能会像这样:

var person1 = {
    name: ‘Mike‘,
    age: 5,
    friends: [ ‘Jack‘, ‘Candy‘ ]
};

对于这个例子,如果我们想实现深拷贝,就需要遍历person1的所有属性。由于name、age是基本类型,所以直接赋值没有问题,但是friends是数组,也就是引用类型,那么我们也需要遍历friends,万一friends数组的某个元素是引用类型,那么继续,以此类推......

所以,实现深拷贝并不是那么容易的,下面我们来看看jQuery中是如何处理的。

// 这是一个用来扩展jQuery的方法,jQuery.extend([boolean], arg1, arg2, ...)
// 第一个参数可选,表示进行浅拷贝或深拷贝,默认false
// 第二个参数是拷贝后的存放目标(后面简称拷贝目标)
// 第三个及以后的参数都是拷贝对象(后面简称拷贝对象)

jQuery.extend = jQuery.fn.extend = function () {
        var options, name, src, copy, copyIsArray, clone,
            target = arguments[0] || {},
            i = 1,
            length = arguments.length,
            deep = false;

        if (typeof target === "boolean") {
            deep = target;

            target = arguments[i] || {};
            i++;
        }

        if (typeof target !== "object" && !jQuery.isFunction(target)) {
            target = {};
        }

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

        // 以下开始重点
        // 遍历arg2, arg3, ...

        for (; i < length; i++) {

            // 如果拷贝对象不是null,并存放到变量options中
            if ((options = arguments[i]) != null) {

                // 遍历拷贝对象的属性
                for (name in options) {
                    src = target[name]; // 拷贝目标中的相应属性值
                    copy = options[name]; // 拷贝对象中的相应属性值

                    // 防止循环引用
                    if (target === copy) {
                        continue;
                    }

                    // 如果deep(布尔参数)为true,也就是要进行深拷贝,并且拷贝对象中该属性值是数组或者对象
                    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 : {};
                        }

                        // 以该属性值作为拷贝对象,递归调用jQuery.extend()
                        target[name] = jQuery.extend(deep, clone, copy);

                    // 否则浅拷贝
                    } else if (copy !== undefined) {
                        target[name] = copy; // 直接赋值
                    }
                }
            }
        }

        // 返回最终的拷贝目标
        return target;
    };

 

欢迎补充和指正

 

以上是关于Javascript中的浅拷贝和深拷贝的主要内容,如果未能解决你的问题,请参考以下文章

javascript中的浅拷贝和深拷贝

JavaScript学习(七十八)—实现对数据的浅拷贝和深拷贝

python中的浅拷贝和深拷贝

js对象浅拷贝和深拷贝详解

C#的浅拷贝和深拷贝

python中的浅拷贝和深拷贝