前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >js库 - 浅拷贝 & 深拷贝

js库 - 浅拷贝 & 深拷贝

作者头像
xing.org1^
发布2018-12-25 14:57:25
2.1K0
发布2018-12-25 14:57:25
举报
文章被收录于专栏:前端说吧前端说吧

学了堆栈内存空间,应该就理解了什么叫简单数据类型存在栈内存,复杂数据类型存在堆内存了。

然后面试中经常会问、业务中也经常会遇到的问题就是深浅拷贝的问题了。

栈内存中简单数据类型直接拷贝就能得到一个副本,但是复杂数据类型的拷贝如果也想得到一个副本,就需要深拷贝了。

浅拷贝:

代码语言:javascript
复制
var  a = 1;

var b = a;

这就是浅拷贝了,虽然你视觉上看上去a = b;但是修改b的值,a不会收影响。因为b是a的一个副本,就像你拷贝了一个文件夹副本一样。修改副本,源文件夹不会受影响。

但这种拷贝情况只局限在简单类型的拷贝:

string、number、boolean、null、undefiend

如果你拷贝一个数组/对象(以数组为例):

代码语言:javascript
复制
var c = [1,2,3];

var d = c;

d.push(4);

console.log(c)

c也被改变了。

但是如果你改变d的整个数据,让他等于一个新的数组(甚至对象)c这时返倒不受影响了。

很奇怪很费解吧?

这是因为:

数组、对象这类复杂类型数据结构,在栈内存里存放的只是指向堆内存中存放数据的地址,

你直接d = c; 拷贝的也是一个副本,但这个副本区别之处是,他并非数据的副本,而是栈内存地址的副本。

这个副本d的地址改变,对c没有影响,而你给d重新指向一个新的数组,就是改变地址了。此时只有d指向了新数组5,6,7,但是c不受影响,所以打印出来的不变。

但是虽然有两个地址,一个是原、一个是副本。可他们同时指向同一个堆内存的数据。

这样看来,你拷贝出来的d和c用的是同一个数组。

所以d.push执行以后,并不是c也跟着push了,而是c指向的数组和被d.push的是同一个数组。

你可以想象成,同一个房间,d和c都拿着钥匙,这俩钥匙是不同的,但是他们都是同一间屋子的钥匙。d用钥匙打开门往屋子里放了一个东西。那么c打开这个屋子,这个东西也还会在。

以上这些,就是基础知识的解读。

深拷贝

具体深拷贝就是要理解了复杂类型拷贝的缺点,然后再进行弥补。

既然想要复杂类型也像简单类型那样拷贝一个新数据的话,就不单单是拷贝地址了。而是要重新建立一个新数据存放空间,然后挨个把想要拷贝的东西搬进来才行。

所以拷贝前我们得判断一个数据他到底是什么类型的,好“对症下药”:

代码语言:javascript
复制
getType: function (target) {
  /*
    * @Author: guojufeng@ 
    * @Date: 2017-12-20 15:07:06
    * @purpose 获取一个值的类型
    * @param {variateName} target: 要获取类型的变量名或对象
    * @output {string} result || "null": 返回的参数 - result或者null,为字符串形式的
    */
  if (target === null) {
    console.log("getType类型判断为: " + target)
    return "null";
  }
  let result = typeof (target);
  if (result == "object") {
    if (target instanceof Array) {
      result = "array";
    } else if (target instanceof Object) {
      let target = Object.prototype.toString.call(target);
      if (target == "[object Object]") {
        result = "object";
      } else {
        result = target;//构造类
      }
    }
  }
  console.log("getType类型判断为: " + result)
  return result; //返回类型值的字符串形式
}

以上代码,判断一个值的类型,输入值本身,返回字符串形式的类型描述。

如果是简单类型,返回typeof方法返回的对应值即可。

这里特殊处理了null,因为他用typeof返回object。

然后对于复杂类型的数据,再深入判断其实array类型还是object类型。

对object类型中,还有构造类需要区分。直接返回object String这样类型的。但其实在深拷贝阶段,直接将其放到object形式处理了。

然后就是深拷贝的代码:

代码语言:javascript
复制
deepClone: function (origin) {
  /*
    * @Author: guojufeng@ 
    * @Date: 2018-10-30 20:48:44 
    * @purpose 深度克隆
    * @param {variateName} origin: 要克隆的对象变量名
    * @output {对应值} 根据origin的类型返回的具体值
    */
  let type = this.getType(origin),
    target;
  if (type == "array") {
    target = [];
    /* 数组 */
    origin.forEach(el => {
      // console.log("ele",el)
      target.push(this.deepClone(el));
    });
  } else if (type == "object") {
    /* 对象 */
    target = {};
    for (const key in origin) {
      if (origin.hasOwnProperty(key)) {
        /* 注意,只拷贝元素身上的,而不拷贝其原型上的值 */
        const el = origin[key];
        target[key] = this.deepClone(el);
      }
    }
  } else if (type == "function") {
    /* 函数 */
    target = function () {};
    target = origin;
  } else {
    /* 原始值 */
    target = origin;
  }
  return target;
}

对于简单类型,直接进行拷贝

对于函数,新建一个function,然后拷贝

对于数组,新建一个数组,然后 forEach 遍历拷贝。如果循环过程中,数组中嵌套复杂类型,再次递归调用深拷贝方法。

对于对象,新建一个对象,然后for in遍历拷贝非原型值。如果循环过程中,对象中嵌套复杂类型,再次递归调用深拷贝方法。

以上,就是整个逻辑。

声明:

  请尊重博客园原创精神,转载或使用图片请注明:

  博主:xing.org1^

  出处:http://www.cnblogs.com/padding1015/

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018-12-10 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 浅拷贝:
  • 深拷贝
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档