专栏首页西安-晁州js中的深浅拷贝

js中的深浅拷贝

js中的深浅拷贝

js中有深拷贝、浅拷贝一说,所谓的深浅拷贝是针对value类型为引用类型(函数、对象、数组)而言的,大概理解的就是:

浅拷贝: 拷贝出的对象c和原始对象o,c和o在key对应的value为引用类型时,其指向同一块内存地址,修改一个必然影响另一个。

举个浅拷贝的例子:

var shallowCopy = function (src) {
  var dst = {}
  for (const key in src) {
    if (src.hasOwnProperty(key)) {
      dst[key] = src[key];
    }
  }
  return dst
}

var obj = { a: 1, arr: [2, 3] }
var shallowObj = shallowCopy(obj)
console.log(shallowObj) // { a: 1, arr: [ 2, 3 ] }

// 浅拷贝指向同一地址,修改shallowObj的arr则obj.arr也被修改,a属性是值类型不存在该问题
shallowObj.arr[1] = 5;
console.log(obj)   // { a: 1, arr: [ 2, 5 ] }
shallowObj.a = 22

// { a: 1, arr: [ 2, 5 ] } { a: 22, arr: [ 2, 5 ] }
console.log(obj, shallowObj)   

var obj2 = { a: 1, t: { b: 3, c: 4, d: { e: 6 } } }
var shallowObj2 = shallowCopy(obj2)
// { a: 1, t: { b: 3, c: 4, d: { e: 6 } } }
console.log(shallowObj2)

深拷贝: 深拷贝出的对象和原对象是完全分开的内存地址,不存在修改一个也修改了另一个的问题。

深拷贝的简单实现:

var china = {
  nation: '中国',
  birthplaces: ['北京', '上海', '广州'],
  skincolr: 'yellow',
  friends: ['sk', 'ls']
}
//深复制,要想达到深复制就需要用递归
function deepCopy(o, c) {
  var c = c || {}
  for (var i in o) {
    if (typeof o[i] === 'object') {
      //要考虑深复制问题了
      if (o[i].constructor === Array) {
        //这是数组
        c[i] = []
      } else {
        //这是对象
        c[i] = {}
      }
      deepCopy(o[i], c[i])
    } else {
      c[i] = o[i]
    }
  }
  return c
}
var result = { name: 'result' }
result = deepCopy(china, result)
result.friends.push('slj')
/**
{ nation: '中国',
  birthplaces: [ '北京', '上海', '广州' ],
  skincolr: 'yellow',
  friends: [ 'sk', 'ls' ] }
{ name: 'result',
  nation: '中国',
  birthplaces: [ '北京', '上海', '广州' ],
  skincolr: 'yellow',
  friends: [ 'sk', 'ls', 'slj' ] }
 */
console.dir(china)
console.dir(result)

其次,深拷贝使用JSON.parse(JSON.stringify(src))也可以简单实现,只是该实现破坏了原型链;lodash、jquery提供了相应的工具函数来实现深浅拷贝,就不记录了。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • mybatis学习之动态sql

    mybatis的动态sql语句很强大,在mapper映射文件中使用简单的标签即可实现该效果,下面一个个记录: 1、select查询 简单的select类似如下:...

    用户1141560
  • nodejs图片上传

    node中图片上传的中间键很多,比如formidable等,这里我们使用nodejs中的fs来实现文件上传处理: 1、安装中间键connect-multipar...

    用户1141560
  • css布局记录之双飞翼布局、圣杯布局

    双飞翼布局和圣杯布局是比较常用的布局方式,都是为了实现一行三列,并且两侧列固定宽度,中间列宽度自适应的效果:直接上代码记录下:

    用户1141560
  • 2000! | 看上去如此简单的面试题,让太多“前端”英雄好汉折戟

    HTML5学堂-码匠:求某个数字的阶乘,很难吗?看上去这道题异常简单,却不曾想里面暗藏杀机,让不少前端面试的英雄好汉折戟沉沙。 面试真题题目 如何求“大数”的阶...

    HTML5学堂
  • Ghidra:这个来自NSA的软件逆向工程工具终于来啦!

    是的,大家没有看错!Ghidra完全出自NSA之手,它是一款软件逆向工程(SRE)框架,由美国国家安全局研究理事会负责开发、升级和维护。这款框架包含了一整套功能...

    FB客服
  • 学习笔记CB011:lucene搜索引擎库、IKAnalyzer中文切词工具、检索服务、word2vec

    影视剧字幕聊天语料库特点,把影视剧说话内容一句一句以回车换行罗列三千多万条中国话,相邻第二句很可能是第一句最好回答。一个问句有很多种回答,可以根据相关程度以及历...

    利炳根
  • Java并发-21.ConcurrentLinkedQueue

    获取头节点的元素,然后判断头结点是否为空,为空说明其他线程已经获取了该节点,如果不为空,CAS将头结点设置为null,成功返回头结点,不成功重新获取。

    悠扬前奏
  • ConcurrentLinkedQueue 源码分析

    ConcurrentLinkedQueue 是一个线程安全且 非阻塞 的 无界 队列,它采用先进先出的规则,实现了 AbstractQueue 基础抽象类和 Q...

    itliusir
  • DCIG全闪存购买指南新鲜出炉,华为OceanStor 全闪存为何能占据C位?

    全球著名技术分析机构DCIG发布了最新的《DCIG 2020年-2021年全闪存阵列购买指南》。不出意外,华为OceanStor全闪存占据C位,位列最佳推荐榜首...

    大数据在线
  • 2018年全闪存阵列市场七大趋势

    全球著名IT产品与技术分析机构DCIG近日给出了全闪存阵列(AFA,All-Flash Array)市场的七大趋势,DCIG分析师认为这七大重要的市场趋势将影响...

    大数据在线

扫码关注云+社区

领取腾讯云代金券