美文网首页
深拷贝与浅拷贝

深拷贝与浅拷贝

作者: 弹指一挥间_e5a3 | 来源:发表于2019-05-21 17:01 被阅读0次

    什么是深拷贝与浅拷贝?

    就是假设B复制了A,当修改A时,看B是否会发生变化,如果B也跟着变了,说明这是浅拷贝。如果B没变,那就是深拷贝,自食其力。

    例如let a = [0,1,2,3,4],b=a复制,修改a[0]=1,此时数组b的值也发生了变化,这就是浅拷贝。深拷贝就是修改a的值,而b的值不发生变化。


    image.png

    为什么?

    那么这里,就得引入基本数据类型与引用数据类型的概念了。

    面试常问,基本数据类型有哪些,number,string,boolean,null,undefined,symbol以及未来ES10新增的BigInt(任意精度整数)七类。

    引用数据类型(Object类)有常规名值对的无序对象{a:1},数组[1,2,3],以及函数等。

    而这两类数据存储分别是这样的:

    a.基本类型--名值存储在栈内存中,例如let a=1;

    image

    当你b=a复制时,栈内存会新开辟一个内存,例如这样:

    image

    所以当你此时修改a=2,对b并不会造成影响,因为此时的b已自食其力,翅膀硬了,不受a的影响了。当然,let a=1,b=a;虽然b不受a影响,但这也算不上深拷贝,因为深拷贝本身只针对较为复杂的object类型数据。

    b.引用数据类型--名存在栈内存中,值存在于堆内存中,但是栈内存会提供一个引用的地址指向堆内存中的值,我们以上面浅拷贝的例子画个图:

    image

    当b=a进行拷贝时,其实复制的是a的引用地址,而并非堆里面的值。

    image

    而当我们a[0]=1时进行数组修改时,由于a与b指向的是同一个地址,所以自然b也受了影响,这就是所谓的浅拷贝了。

    image

    那,要是在堆内存中也开辟一个新的内存专门为b存放值,就像基本类型那样,岂不就达到深拷贝的效果了

    image

    实现深拷贝的三种方法

    1.递归

    function deepClone(obj) {
        var newObj = obj instanceof Array ? [] : {};
        //obj属于基本数据类型,直接返回obj
        if(typeof obj !== 'object') {
            return obj;
        } else {
        //obj属于数组或对象,遍历它们
            for(var i in obj) {
                newObj[i] = typeof obj[i] === 'object' ? deepClone(obj[i]):obj[i]; 
            }
        }
        return newObj;
    }
    

    2.JSON对象的parse和stringify

    function deepClone(obj){
        let _obj = JSON.stringify(obj),
            objClone = JSON.parse(_obj);
        return objClone
    }   
    

    3.jQuery的extend方法。

    let a=[0,1,[2,3],4],
        b=$.extend(true,[],a);
    

    相关文章

      网友评论

          本文标题:深拷贝与浅拷贝

          本文链接:https://www.haomeiwen.com/subject/klvjzqtx.html