JS浅克隆与深克隆

为什么会有克隆这么个玩意呢?

上代码:

var a = [1,2,3];
var b = a;
a[1] = 4;
console.log(b[1]);  // 4

这就是我们为什么提到克隆的原因。我只不过改了a数组的内容,怎么b数组的内容也跟着一起变了嘞?这里需要知道ab在栈和堆中的存储情况

看图:

JS栈堆中数组的存储情况

大概说下,数组的内容是存储在堆中的,而ab不过是作为一个引用(也可以参考C语言的指针)指向了数组内容的存储地址,上面var b = a;只不过是将a引用赋值给了b,并不是将数组内容拷贝一份出来,然后让b指向新拷贝出来的数组内容。所以a或者b修改数组内容,其实都是修改的同一个。

又因为基本变量(js的原始值undefinednullnumberstringboolean类型)一般都会直接放在栈中,因此克隆只针对对象数组函数等复杂数据。

浅克隆

上面栗子就是浅克隆,说白了就是只复制引用地址,不复制内容。实现方法如上栗子。

深克隆

那当然就是连同内容一起复制啦。

看代码:

function deepClone(val) {
    // 如果不是对象类型,那么就直接返回
    if (typeof val != "object") {
        return val;
    }
    var result = {};
    for (var i in val) {
        result[i] = deepClone(val[i]);  // 递归循环完成深克隆
    }
    return result;
}

将刚才那个栗子中修改一下

var a = [1,2,3];
var b = deepClone(a);
a[1] = 4;
console.log(b[1]);  // 2

然后我们就可以发现这个结果已经是我们想要的了。

那么现在就让我们偷个懒,要是每次都要写这么个函数也挺累的。

我们再把刚才的那个栗子修改一下,这次不使用深克隆函数了。

var a = [1,2,3];
var b = JSON.parse(JSON.stringify(a));
a[1] = 4;
console.log(b[1]);  // 2

诶!一行代码解决了耶。

这里的原理在于将对象转换成JSON字符串后又将JSON字符串重新生成为一个一模一样的新的对象,于是就间接地帮我们完成了深克隆的工作。

总结

深浅克隆只要理解了堆栈其实就是这么一回事的东西,没啥大不了的 _(:з」∠)_

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值