热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

js浅拷贝直接赋值_JS深拷贝与浅拷贝

Intro首先明确两个概念:深拷贝和浅拷贝只针对像Object,Array这样的复杂对象的,如果只是值类型数据,则不存在所谓的深拷贝&#x
Intro

首先明确两个概念:

  • 深拷贝和浅拷贝只针对像 Object, Array 这样的复杂对象的,如果只是值类型数据,则不存在所谓的深拷贝;

  • 浅拷贝只拷贝一层对象的属性,而深拷贝则递归拷贝了所有层级。

这就引出了另一个基础的概念:数据类型。

数据类型

通常把数据类型分为 值类型 和 引用类型:

  • 值类型(基本类型):字符串(String)、数字(Number)、布尔(Boolean)、对空(Null)、未定义(Undefined)、Symbol。

  • 引用数据类型:对象(Object)、数组(Array)、函数(Function)。

注:Symbol 是 ES6 引入了一种新的原始数据类型,表示独一无二的值。

以Number类型举例

var a = 1;
var b = a;    // b: 1
b = 2;
console.log(a, b);    // a: 1, b: 2

可以看出基本类型并没有什么问题,接下来看引用数据类型的例子:

var obj1 = {id: '1'};
var obj2 = obj1;    // [object Object] { id: 1 }
obj2.id = 2;
console.log(obj1, obj2);    // [object Object] { id: 2 }, [object Object] { id: 2 }

可以看到,当修改 obj2的id属性时,obj1的id属性也同时修改了,这是因为 Javascript 存储对象都是存在内存里的一块地址,而简单的浅拷贝或者赋值只是把对象的引用(或者说指针)拷贝给了另一个对象,无论哪个变量修改,其实操作的都是内存里的同一个对象

【补】最新的 ECMAScript 标准定义了 8 种数据类型:

  • 7种原始类型:Boolean、Null、Undefined、Number、BigInt、String、Symbol ;

  • 和Object

接着深究,就牵涉到了栈和堆的概念。

栈和堆

栈指的是栈内存(stack):

Javascript中的基本类型按值存在栈内存中,数据的大小都是确定或者是在一定范围内的,由系统自动分配和释放,所以更容易对内存进行管理;

堆指的是堆内存(heap):

Javascript中引用类型如Object、Array、Function等存在于堆中,指针存放在栈中。当访问引用类型时,先在栈中找到对应的地址指针,然后根据指针找到堆中的数据。

edd0256632687e2835f5ff8594a854dd.png

底层原理清楚后,就更能理解数据类型的一些行为了。

实现

明白了数据类型以及栈和堆,就能很容易的理解深拷贝和浅拷贝:

  • 浅拷贝只会将对象的各个属性进行依次复制,并不会进行递归复制;

  • 深拷贝不仅将原对象的各个属性逐个复制出去,而且将原对象各个属性所包含的对象也依次采用深复制的方法递归复制到新对象上。

浅拷贝

方法一:遍历被拷贝对象的属性

function copy(obj) {
  var copyObj = {};
  for(var key in obj) {
    copyObj[key] = obj[key];
  }
  return copyObj;
}
var o1 = {name: 'o1'}
var o2 = copy(o1);
console.log(o2);    // [object Object] { name: "o1" }

方法二:Object.assign

Object.assign() 方法用于将所有可枚举属性的值从一个或多个源对象分配到目标对象。它将返回目标对象。

var source = {success: true, count: 1};
var target = Object.assign({}, source);    // [object Object] { count: 1, success: true }

深拷贝

方法一:JSON.parse(JSON.stringify(data))

var source = {success: true, content: {isUserLogin: true}};
var target = JSON.parse(JSON.stringify(source));    // {success: true, content: {isUserLogin: true}}

但是序列化有很多问题:

  • 如果对象中有函数或undefined,则有可能将函数丢失;

  • 如果对象中有NaN、Infinity、-Infinity,则变成null;

  • 如果对象中有Date、Error等对象,则会变成空对象。

如果确定对象中的数据是“常规”的,则序列化是最简单实现深拷贝的方法。

方法二:递归

function deepClone(source) {
  if (source && typeof source === 'object') {
    var target = source instanceof Array ? [] : {};
    // var target = Object.prototype.toString.call(source) === '[object Array]' ? [] : {};
    for (var k in source) {
      if (source.hasOwnProperty(k)) {
        if (source[k] && typeof source[k] === 'object') {
          target[k] = deepClone(source[k]);
        } else {
          target[k] = source[k];
        }
      }
    }
    return target;
  } else {
    return source;
  }
}

  • 首先判断是否为对象,如果不是,则直接返回;

  • 如果是对象,则判断源对象是数组还是对象,来初始化target;

  • 遍历源对象,如果熟悉是基本类型,则直接赋值,否则就递归调用deepClone函数自身;

接下来可优化的点:

  1. 如果是Function、Date、RegExp等类型怎么处理;

  2. 如果对象自身的属性指向自身怎么处理;

  3. 考虑遍历的性能;

第三方库的实现

jQuery

$.clone$.extend

  • 在 jQuery 中有一个叫 $.clone() 的方法,可是它并不是用于一般的 JS 对象的深拷贝,而是用于 DOM 对象,无需关注。

  • 可以通过添加一个参数来实现递归extend。调用$.extend(true, {}, …)就可以实现深拷贝。

var obj1 = {success: true, content: {isLogin: true}};
var obj2 = $.extend({}, obj1);    // 浅拷贝
var obj3 = $.extend(true, {}, obj1);    // 深拷贝

Lodash

_.clone浅拷贝,_.clone(obj, true) 等价于深拷贝_.cloneDeep(obj)

Outro

考察的知识点

  1. 栈和堆的概念;

  2. 基本数据类型和引用数据类型;

  3. 递归思想;




推荐阅读
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 本文介绍了如何在给定的有序字符序列中插入新字符,并保持序列的有序性。通过示例代码演示了插入过程,以及插入后的字符序列。 ... [详细]
  • React项目中运用React技巧解决实际问题的总结
    本文总结了在React项目中如何运用React技巧解决一些实际问题,包括取消请求和页面卸载的关联,利用useEffect和AbortController等技术实现请求的取消。文章中的代码是简化后的例子,但思想是相通的。 ... [详细]
  • 本文介绍了如何使用JSONObiect和Gson相关方法实现json数据与kotlin对象的相互转换。首先解释了JSON的概念和数据格式,然后详细介绍了相关API,包括JSONObject和Gson的使用方法。接着讲解了如何将json格式的字符串转换为kotlin对象或List,以及如何将kotlin对象转换为json字符串。最后提到了使用Map封装json对象的特殊情况。文章还对JSON和XML进行了比较,指出了JSON的优势和缺点。 ... [详细]
  • 本文介绍了一个React Native新手在尝试将数据发布到服务器时遇到的问题,以及他的React Native代码和服务器端代码。他使用fetch方法将数据发送到服务器,但无法在服务器端读取/获取发布的数据。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 使用在线工具jsonschema2pojo根据json生成java对象
    本文介绍了使用在线工具jsonschema2pojo根据json生成java对象的方法。通过该工具,用户只需将json字符串复制到输入框中,即可自动将其转换成java对象。该工具还能解析列表式的json数据,并将嵌套在内层的对象也解析出来。本文以请求github的api为例,展示了使用该工具的步骤和效果。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • JDK源码学习之HashTable(附带面试题)的学习笔记
    本文介绍了JDK源码学习之HashTable(附带面试题)的学习笔记,包括HashTable的定义、数据类型、与HashMap的关系和区别。文章提供了干货,并附带了其他相关主题的学习笔记。 ... [详细]
  • 本文介绍了在iOS开发中使用UITextField实现字符限制的方法,包括利用代理方法和使用BNTextField-Limit库的实现策略。通过这些方法,开发者可以方便地限制UITextField的字符个数和输入规则。 ... [详细]
  • 如何查询zone下的表的信息
    本文介绍了如何通过TcaplusDB知识库查询zone下的表的信息。包括请求地址、GET请求参数说明、返回参数说明等内容。通过curl方法发起请求,并提供了请求示例。 ... [详细]
  • 图像因存在错误而无法显示 ... [详细]
  • 本文介绍了Java后台Jsonp处理方法及其应用场景。首先解释了Jsonp是一个非官方的协议,它允许在服务器端通过Script tags返回至客户端,并通过javascript callback的形式实现跨域访问。然后介绍了JSON系统开发方法,它是一种面向数据结构的分析和设计方法,以活动为中心,将一连串的活动顺序组合成一个完整的工作进程。接着给出了一个客户端示例代码,使用了jQuery的ajax方法请求一个Jsonp数据。 ... [详细]
author-avatar
WLII庾斌_787
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有