JS - 浅拷贝与深拷贝的理解以及简单实现方法
前几天撸项目代码时, 由一个技术点间接牵扯出了这东西. 所以就来总结一下.
深拷贝
拷贝对象每个层级的属性.
作用的对象是 js中引用类型的对象,基本类型没有涉及.
本质上将引用类型的对象在堆上重新开辟一块新的空间进行存放.
var p_1 = {name: '病猫', age: 22};
var p_3 = {name: p_1.name, age: p_1.age};
p_3.name = "迪迦";
console.log("p_1", p_1); //p_1 {name: "子川", age: 22}
console.log("p_3", p_3); //p_3 {name: "迪迦", age: 22}
浅拷贝
对于引用类型的对象, 浅拷贝是在栈上重新定义了一个指针,指向的位置却依然是堆中同一对象.
var p_1 = {name: '病猫', age: 22};
var p_2 = p_1;
p_2.name = "子川";
console.log("p_1", p_1); //p_1 {name: "子川", age: 22}
console.log("p_2", p_2); //p_2 {name: "子川", age: 22}
实现方法
①: JSON的序列化与反序列化
let o = JSON.parse(JSON.stringify(obj));
//但是obj中若含有 function, undefined, 正则表达式 时,
//o 中 function, undefined不存在, 正则表达式对应变成了 {};
②: 递归 (对日期和正则做了简单处理,)
function deepHandle(obj) {
let tempObj = Array.isArray(obj) ? [] : {};
if (obj && typeof obj === "object") {
for (key in obj) {
if (obj.hasOwnProperty(key)) { //判断自身属性
if (obj[key] && typeof obj[key] === "object") { //是对象类型
let type = Object.prototype.toString.call(obj[key]).slice(8, -1); //具体对象类型
console.log("类型", type);
//Date 与 RegExp 不处理的话, 拷贝之后会成 {},
if (type === 'Date') {
tempObj[key] = new Date(obj[key]);
} else if (type === 'RegExp') {
tempObj[key] = new RegExp(obj[key]);
} else {
tempObj[key] = deepHandle(obj[key]);
}
} else {
tempObj[key] = obj[key];
}
}
}
}
return tempObj;
}
let obj = {
a: '于康',
b: {
age: 21,
city: "山西",
n: [1,2,3, new Date(), /234/igm]
},
c: undefined,
d: null,
e: true,
f: new Date(),
h: {},
i: /234/igm,
toEat: function() {
console.log("吃法")
}
}
let deepObj = deepHandle(obj);
let deepObj2 = JSON.parse(JSON.stringify(obj));
deepObj.i = /123/igm;
console.log("obj", obj);
console.log("deepObj", deepObj);
console.log("deepObj2", deepObj2);
补充一下数据类型
基本数据类型 (number, string, boolean, null, undefined).
①:按值访问,即可操作保存在变量中的实际值.
②:值是不可变的,
③:名-值存储在栈内存中.
let name = "病猫"; name = "迪迦"; //看似 name 的值发生了改变, 实际改变的是 指针指向;
引用数据类型 (即对象类型Object type. 如: Object, Array, Function, Date等)
let obj = {name: "病猫", age: 22}
①: 堆内存中存放引用数据类型的对象 (即 {name: "病猫", age: 22})
②: 栈内存中存放引用数据类型的变量名 (即 obj) 和 此对象的 引用地址(即 指针);
③: javascript 不允许直接访问与操作堆内存空间, 只能操作对象在栈中的引用地址
结尾
以上内容是记录自己理解知识后的笔记, 若理解有误, 望您能指出. 感激不尽. 对于这个知识点,网上很多写的通俗易懂的文章(图文), 很Nice的.
JS - 浅拷贝与深拷贝的理解以及简单实现方法的更多相关文章
- 关于js浅拷贝与深拷贝的理解
前端开发中,一般情况下,很少会去在意深拷贝与浅拷贝的关系. 大家知道,js变量有2种数据类型:基本类型和引用类型.基本类型的拷贝是将整个值完全拷贝一份的,也就是深拷贝.就是开辟了新的堆内存.所以基本类 ...
- 理解js浅拷贝和深拷贝
理解深拷贝和浅拷贝之前先了解下js中的基本类型和引用类型 1.基本类型: 在js中,数据的基本类型undefined,null,string,number,boolean,在变量中赋的实际值,基本类型 ...
- 关于JS浅拷贝和深拷贝
在 JS 中有一些基本类型像是Number.String.Boolean,而对象就是像这样的东西{ name: 'Larry', skill: 'Node.js' },对象跟基本类型最大的不同就在于他 ...
- Java 浅拷贝和深拷贝的理解和实现方式
Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去.举例说明:比如,对象A和对象B都属于类S,具有属性a和b.那么对对象A进行拷贝 ...
- 【转】Java 浅拷贝和深拷贝的理解和实现方式
Java中的对象拷贝(Object Copy)指的是将一个对象的所有属性(成员变量)拷贝到另一个有着相同类类型的对象中去.举例说明:比如,对象A和对象B都属于类S,具有属性a和b.那么对对象A进行拷贝 ...
- c#中浅拷贝和深拷贝的理解
c#中拷贝有浅拷贝和深拷贝之分. 例如对象A,其中有值类型字段和引用类型字段: 1.浅拷贝: 对于值类型字段,直接逐位复制到新拷贝的副本对象中,修改副本的字段的值,不会影响源对象中字段的值: 对于引用 ...
- 浅谈Javascript 浅拷贝和深拷贝的理解
javascript中存储对象都是存地址的. 浅拷贝:浅拷贝是都指向同一块内存区块,浅拷贝共用同一内存地址,你改值我也变.如果拷贝的对象里面的值是一个对象或者数组,它就是浅拷贝,拷贝的知识引用地址. ...
- js 浅拷贝和深拷贝
传值与传址 了解了基本数据类型与引用类型的区别之后,我们就应该能明白传值与传址的区别了.在我们进行赋值操作的时候,基本数据类型的赋值(=)是在内存中新开辟一段栈内存,然后再把再将值赋值到新的栈中.例如 ...
- 前端【JS】,深拷贝与浅拷贝的区别及详解!
我是前端小白一枚,为了巩固知识和增强记忆,开始整理相关的知识,方便以后复习和面试的时候看看.OK,让我们进入正题~ 先说说浅拷贝和深拷贝的理解吧,个人是这样理解的:两个对象A.B, A有数据B为空,B ...
随机推荐
- BZOJ 2527 [POI2011]MET-Meteors (整体二分+树状数组)
题目大意:略 洛谷传送门 整体二分裸题 考虑只有一个国家的情况如何处理 对询问数量二分答案,暴力$O(m)$打差分,求前缀和验证,时间是$O(mlogK)$ 如果有$n$个国家,就是$O(nmlogK ...
- xunsearch实战经验总结
一.定义好配置文件(非常关键) a):如果需要做精确搜索建议对字段设定index=self,tokenizer = full,不然xunsearch会对字段做分词处理: b):数字区间搜索需设定 ty ...
- MongoDB记录(坑在末尾)
Mongo数据库基本配置 基本配置 密码配置 pymongo认证 参考资料 基本配置 基本配置包括 1.端口号:默认27017,安全性较低 2.数据库文件位置 3.日志文件位置 4.日志写入模式 5. ...
- Charles抓包工具抓取HTTS请求
1. 移动端(手机)配置证书 1.1 进入Charles,点击Help,选择SSL Proxying --> 选择安装手机证书 1.2 在Charles弹框提示中,按照提示内容,进入手机进入下图 ...
- 用chrony代替ntpd时间同步服务器
Chrony是一个开源的自由软件,它能保持系统时钟与时钟服务器(NTP)同步,让时间保持精确. 它由两个程序组成:chronyd和chronyc. chronyd是一个后台运行的守护进程,用于调整内核 ...
- BIRT报表Cannot open the connection for the driver:org.eclipse.birt.report.data.oda.jdbc.dbprofile
现象:报表不能打开: 找了半个小时,随手改了一下tomcat/conf文件夹下的context.xml文件 <Context xmlBlockExternal="false" ...
- shell脚本监测文件变化
1. 我使用过的Linux命令之du - 查看文件的磁盘空间占用情况 用途说明 du命令是用来查看磁盘空间占用情况的,在Linux系统维护时常会用到,并且通常与df命令搭配使用.首先使用df看一下各个 ...
- 推荐几款VisualStudio的插件
继前几天推荐了一款转换vs插件的插件后,借着安装VS2013之际,把我比较喜欢的几个插件继续推荐一下. C# Outline 2013 2013 C#的代码折叠最小只能到函数级,不像C++那样可以折叠 ...
- redis代码解析-事务
redis 的事务相关的几个命令分别为 watch multi exec. watch 可以监控一个变量在事务开始执行之前是否有被修改.使用方式为: WATCH key [key ...] 在redi ...
- 【C/C++多线程编程之十】pthread线程私有数据
多线程编程之线程私有数据 Pthread是 POSIX threads 的简称.是POSIX的线程标准. 线程同步从相互排斥量[C/C++多线程编程之六]pthread相互排 ...