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 条评论
登录 后参与评论

相关文章

来自专栏web前端

JavaScript基础学习--13字符串、查找高亮显示

Demos:   https://github.com/jiangheyan/JavaScriptBase 一、字符串      1、str.length; ...

2136
来自专栏专注 Java 基础分享

漫谈计算机编码

我们知道,在计算机内部,所有的信息都是以二进制形式进行存储。无论是字符,或是视频音频文件,最终都会对应到一串由 0 和 1 构成的数字串。所以从我们能看懂的人类...

3976
来自专栏小樱的经验随笔

ASCII,Unicode和UTF-8

一、ASCII码 我们知道,计算机内部,所有信息最终都是一个二进制值。每一个二进制位(bit)有0和1两种状态,因此八个二进制位就可以组合出256种状态,这被称...

30212
来自专栏xx_Cc的学习总结专栏

iOS底层原理总结 - 探寻Runtime本质(二)

2532
来自专栏机器学习算法与Python学习

笔记 | 史上最全的正则表达式

原文:http://www.cnblogs.com/zxin/archive/2013/01/26/2877765.html 点击文末阅读原文即可 很多不太懂正...

32511
来自专栏Python攻城狮

正则表达式1.正则表达式概述2.re模块操作3.表示字符4.re模块的高级用法5.贪婪和非贪婪

在Python中需要通过正则表达式对字符串进行匹配的时候,可以使用一个模块,名字为re

1922
来自专栏数据结构与算法

病毒

【问题描述】   有一天,小y突然发现自己的计算机感染了一种病毒!还好,小y发现这种病毒很弱,只是会把文档中的所有字母替换成其它字母,但并不改变顺序,也不会增加...

5197
来自专栏黑泽君的专栏

【Java面试复习经典】传智播客Java就业班入学测试题及答案解析(2012年版)

  共50道题,每道题2分,总分100分,80分为合格。   注意,题目有多选,也有单选。请认真作答。

1513
来自专栏PHP实战技术

PHP常用正则表达式大全

  匹配数字   "^\d+$" //非负整数(正整数 + 0)   "^[0-9]*[1-9][0-9]*$" //正整数   "^((-\d+)|(0+))...

4598
来自专栏ml

NYOJ-------表达式求值

时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 Dr.Kong设计的机器人卡多掌握了加减法运算以后,最近又学会了一些简单的函数求...

38910

扫码关注云+社区

领取腾讯云代金券