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

javascript深拷贝与浅拷贝

作者: 小梁姐姐 | 来源:发表于2017-11-04 14:49 被阅读0次

在工作中会遇到各式各样的与深拷贝浅拷贝相关的问题,其实造成这类问题的根本原因是javascript中,不同的数据类型的传值方式不一致,首先我们先说一下js中都有哪些数据类型:

1.javascript的数据类型及传值方式

  • 简单数据类型
    简单数据类型也就是值类型,简单数据类型有:Undefined, Null,Boolean,Number,String.
    传值方式:变量的交换等于在一个新的作用域创建一个新的空间,新空间与之前的空间互不相关和影响。
  • 复杂数据类型
    复杂数据类型也叫引用类型,常见的复杂数据类型有:Object、Array、Function。
    传值方式:变量的交换,并不会创建一个新的空间,而是让对象或方法和之前的对象或方法,同时指向一个原有空间(即一个地址)。就如同原来一个人有家门的钥匙,之后这个人结婚了,就配了一把钥匙给自己的妻子,这时候,两个人共同有家的钥匙,但是家还是一个家。

2.浅拷贝

什么是浅拷贝呢?我们来直接看代码吧!

    var obj = {a: 10, b: 20, c: 30}
    var obj2 = obj
    obj2.b = 50
    console.log(obj) //{a: 10, b: 50, c: 30}
    console.log(obj2) //{a: 10, b: 50, c: 30}

复制一份obj为obj2,修改了obj2.b的值,由于这两个对象所指向同一个地址,所以造成obj.b的值也跟着改变了,这就叫做浅拷贝。

3.深拷贝

我们希望改变复制过来的新值不对旧数据进行修改,这就是深拷贝,那么我们要怎么做才可以深拷贝呢

方法1.Object.assign()
    var obj = {a: 10, b: 20, c: 30}
    var obj2 = Object.assign({}, obj) //第一个参数要为{}
    obj2.b = 50
    console.log(obj) //{a: 10, b: 20, c: 30}
    console.log(obj2) // {a: 10, b: 50, c: 30}

这个方法看起来不错,使用起来也比较方便,但是这个方法的坑不小!只适用于一层的深度拷贝,对于多级的拷贝就歇菜了~例如

    var obj = {a: 10, b: {aa: 'aa', bb: 'bb'}, c: 30}
    var obj2 = Object.assign({}, obj)
    obj2.b.aa = 50
    console.log(obj) //{a: 10, b: {aa: 50, bb: 'bb'}, c: 30}  改变了
    console.log(obj2) // {a: 10, b: {aa: 50, bb: 'bb'}, c: 30}
方法2.JSON.parse(JSON.stringify())
    var obj = {a: 10, b: {aa: 'aa', bb: 'bb'}, c: 30}
    var obj2 = JSON.parse(JSON.stringify(obj))
    obj2.b.aa = 50
    console.log(obj) //{a: 10, b: {aa: 'aa', bb: 'bb'}, c: 30} 没有变没有变
    console.log(obj2) // {a: 10, b: {aa: 50, bb: 'bb'}, c: 30}

嗯嗯,这个方法解决了方法1存在的问题,但是不是就是完美的方法了呢?答案是一个非常肯定的NO!!!!!!,那么这个方法存在什么样的问题呢?首先它会抛弃对象的constructor。也就是深拷贝之后,不管这个对象原来的构造函数是什么,在深拷贝之后都会变成Object。

这种方法能正确处理的对象只有 Number, String, Boolean, Array, 扁平对象,即那些能够被 json 直接表示的数据结构。也就是说,只有可以转成JSON格式的对象才可以这样用,像function没办法转成JSON。

    var obj = {a: 10, b: function(){}, c: 30}
    var obj2 = JSON.parse(JSON.stringify(obj))
    console.log(obj.b) //function(){}
    console.log(obj2.b) //undefine
方法3.slice()或concat()
    var arr = ['a', 'b', 'c', 'd']
    var arr2 = arr.slice(0)
    arr2.push('haha')
    console.log(arr) // ["a", "b", "c", "d"]
    console.log(arr2) // ["a", "b", "c", "d", "haha"]

对数组进行深度克隆可以使用slice()/concat()方法,由上面的例子可以看到这个方法针对一层的深度拷贝是可以的,但是对多级的拷贝的效果如何呢?

    var arr = ['a', 'b', ['aa', 'bb'], 'd']
    var arr2 = arr.slice(0)
    //var arr2 = arr.concat()
    arr2[2].push('haha')
    console.log(arr) // ["a", "b",['aa', 'bb','haha'], "d"]  改变了
    console.log(arr2) // ["a", "b",['aa', 'bb','haha'], "d"]

由上面例子可以看出slice()/concat()方法对多级的拷贝也是没有效果的。

方法4.自定义方法

上面的方法或多或少都有这一些问题,为了使用更加方便,我们可以自定义一个方法deepcope进行深度克隆。

    function deepcope(data) {
        var temp;
        if(data instanceof Array) { //克隆的是数组
            temp = []
            for(var i = 0, l = data.length; i < l; i++){
                temp.push(deepcope(data[i]))
            }
        } else if (data instanceof Object) { //克隆的是对象
            temp = {}
            for(var key in data){
                temp[key] = deepcope(data[key])
            }
        } else {  //克隆的既不是数组也不是对象则返回原数据
            temp = data
        }
        return temp
    }

方法写好了现在我们来试试这个方法的效果如何,首先先尝试一个数组的深度克隆:

    //一层克隆
    var arr = ['a', 'b', 'c']
    var arr2 = deepcope(arr)
    arr2[2]='change'
    console.log(arr2) //['a', 'b', 'change']
    console.log(arr) //['a', 'b', 'c']

    //多层克隆
    var arr = ['a', 'b', [['aaa','bbb'], 'bb'], 'd']
    var arr2 = deepcope(arr)
    arr2[2][0][0]='change'
    console.log(arr2) //['a', 'b', [['change','bbb'], 'bb'], 'd']
    console.log(arr) //['a', 'b', [['aaa','bbb'], 'bb'], 'd']

我们自己写的方法可以完美的进行数组一层以及多层拷贝,那么针对对象的拷贝效果如何呢?我们再次尝试一下:

    //一层克隆
    var obj = {a:'a', b:'b', c:'c'}
    var obj2 = deepcope(obj)
    obj2.b = 'change'
    console.log(obj2)   //{a:'a', b:'change', c:'c'}
    console.log(obj)   //{a:'a', b:'b', c:'c'}

    //多层克隆
    var obj = {a:'a', b:{aa:'aa', bb:{aaa:'aaa'}}, c:'c'}
    var obj2 = deepcope(obj)
    obj2.b.bb.aaa = 'change'
    console.log(obj2)  //{a:'a', b:{aa:'aa', bb:{aaa:'change'}}, c:'c'}
    console.log(obj)  //{a:'a', b:{aa:'aa', bb:{aaa:'aaa'}}, c:'c'}

不错! deepcope方法也可以将对象进行深度拷贝。那么对象和数组的混合效果如何?

    var obj = {a:'a', b:[{aa:'aa', bb: ['aaa', 'bbb']},'bb'], c:'c'}
    var obj2 = deepcope(obj)
    obj2.b[0].bb[0] = 'change'
    console.log(obj2)   // {a:'a', b:[{aa:'aa', bb: ['change', 'bbb']},'bb'], c:'c'}
    console.log(obj)   // {a:'a', b:[{aa:'aa', bb: ['aaa', 'bbb']},'bb'], c:'c'}

嗯!由以上示例我们可以得出结论:我们自定义的deepcope方法可以对数组以及对象进行深度克隆!

相关文章

  • 深入理解JavaScript中的堆与栈 、浅拷贝与深拷贝

    JavaScript中的浅拷贝与深拷贝 学了这么长时间的JavaScript想必大家对浅拷贝和深拷贝还不太熟悉吧,...

  • 浅拷贝和深拷贝

    本文参考:JavaScript中的浅拷贝和深拷贝js 深拷贝 vs 浅拷贝深入剖析 JavaScript 的深复制...

  • js浅拷贝和深拷贝

    javaScript的变量类型 javaScript的变量类型基本类型:引用类型: 浅拷贝和深拷贝的区分 浅拷贝浅...

  • 初探浅拷贝&深拷贝

    思考 这个代码为什么具有深拷贝作用 浅拷贝与深拷贝 在JavaScript中,对于Object和Array这类引用...

  • JavaScript深拷贝、浅拷贝

    JavaScript深拷贝、浅拷贝 浅拷贝:浅拷贝只是复制了内存地址,如果原地址中的对象改变了,浅拷贝出来的对象也...

  • JS中的深拷贝与浅拷贝

    知乎:js中的深拷贝和浅拷贝? 掘金: js 深拷贝 vs 浅拷贝 前言 首先深拷贝与浅拷贝只针对 Object,...

  • JavaScript浅拷贝与深拷贝

    定义 浅拷贝(shallow copy):只复制指向某个对象的指针,而不复制对象本身,新旧对象共享一块内存。深拷贝...

  • 浅拷贝与深拷贝(JavaScript)

    一、预备知识 ECMAScript变量包含两种不同数据类型的值:基本数据类型和引用数据类型。基本数据类型:名值存储...

  • javascript深拷贝与浅拷贝

    在工作中会遇到各式各样的与深拷贝浅拷贝相关的问题,其实造成这类问题的根本原因是javascript中,不同的数据类...

  • JavaScript 深拷贝与浅拷贝

    前言 深拷贝和浅拷贝都只针对引用数据类型,浅拷贝会对对象逐个成员依次拷贝,但只复制内存地址,而不复制对象本身,新旧...

网友评论

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

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