js对象克隆
大家都知道,js的对象是引用类型,如果直接var obj2 = obj,obj2和obj是共享同一个对象实体的,这往往不是我们想要的结果。
官方并没有给出通用的对象克隆方法:
我们给出以下几种写法:
1.通用对象克隆:
function clone(obj, hash = new WeakMap()) {
// 解决循环引用
if (hash.has(obj)) {
return hash.get(obj)
}
let temp = null;
if (obj instanceof Array) {
// 特殊处理数组对象类型
temp = []
hash.set(obj, temp)
obj.forEach(item => {
temp.push(clone(item, hash));
})
} else if (obj instanceof RegExp) {
// 特殊处理正则对象类型
const { source, global, ignoreCase, multiline } = obj;
let flags = '';
if (global) flags += 'g';
if (ignoreCase) flags += 'i';
if (multiline) flags += 'm';
temp = new RegExp(source, flags);
} else if (obj instanceof Date) {
// 特殊处理时间对象类型
temp = new Date(obj.getTime());
} else if (typeof obj === 'object') {
// 处理普通对象类型
// 以obj的原型为原型,构造一个新对象
temp = Object.create(obj.__proto__);
hash.set(obj, temp)
} else {
temp = obj;
}
// 任何对象类型,都遍历递归自身的属性
if (typeof obj === 'object') {
for (let [key, val] of Object.entries(obj)) {
temp[key] = clone(val, hash);
}
}
return temp;
}
遇到数组对象类型、正则对象类型、时间对象类型、普通对象类型都需要有自己的特殊,最后他们都需要遍历递归他们自身的属性
原始类型(包括Symbol)纯复制就可以了
函数做的浅拷贝(因为如果拷贝函数,只能用eval这个危险的工具了,所以这里函数仅作浅克隆,lodash的cloneDeep对函数也是浅克隆处理)
推荐阅读:https://juejin.im/post/5b235b726fb9a00e8a3e4e88
ps:这里还有一些坑,无法克隆 Error 对象,无法克隆原型,无法克隆不可枚举的属性.... 不过这个深克隆已经可以覆盖百分之99的场景了!!
2.JSON对象序列化方法
深拷贝,但是有一大堆坑(推荐阅读:https://juejin.im/post/5abb55ee6fb9a028e33b7e0a):
基础的5个大坑:
1. 函数不能拷贝
2. Symbol不能拷贝
4. undefined不能拷贝
5. 正则拷贝后变成普通对象
6. 循环引用的对象会报错
7. 数组的属性丢失
8. 数组里面的undefined会变成null
9. 会抛弃对象的constructor
。。。。估计还有很多我没想到的
let a = { age: undefined, sex: Symbol('male'), jobs: function() {}, name: 'yck' }
let b = JSON.parse(JSON.stringify(a)) console.log(b) // {name: "yck"}
正常使用:
var obj = {a:1,b:2}
var newObj = JSON.parse(JSON.stringify(obj));
3.dom元素的复制——cloneNode
let div = document.getElementById('box');
let box2 = div.cloneNode(true);
4.es6新方法——Object.assign
浅拷贝
var obj = {a:1,b:2}
var newObj = Object.assign({}, obj);
5. MessageChannel的方法
异步的深克隆,但是无法克隆function 、Symbol
let obj = {a: 1,b: 2}
let {port1, port2} = new MessageChannel();
port2.onmessage = ev => console.log(ev.data)
port2.postMessage(obj)
总结:日常使用还是推荐用lodash的cloneDeep
js对象克隆的更多相关文章
- js 对象克隆方法总结(不改变原对象)
1.通用对象克隆: function clone(obj){ let temp = null; if(obj instanceof Array){ temp = obj.concat(); }else ...
- js对象克隆, 深复制.
亲测有效: //对象克隆 function clone(obj) { // Handle the 3 simple types, and null or undefined if (null == o ...
- js对象克隆方法
方法1: function clone(obj){ var o; switch(typeof obj){ case 'undefined': break; case 'string' : o = ob ...
- js对象的深度克隆
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- js深度克隆对象
js深度克隆对象 js深度克隆对象简单的记录一下,如下代码: var obj = { typeOf: function(obj) { const toString = Object.prototype ...
- js对象简单、深度克隆(复制)
javascript的一切实例都是对象,只是对象之间稍有不同,分为原始类型和合成类型.原始类型对象指的是字符串(String).数值(Number).布尔值(Boolean),合成类型对象指的是数组( ...
- JS对象复制
在JavaScript很多人复制一个对象的时候都是直接用"=",因为大家都觉得脚本语言是没有指针.引用.地址之类的,所以直接用"="就可以把一个对象复制给另外一 ...
- js对象详解(JavaScript对象深度剖析,深度理解js对象)
js对象详解(JavaScript对象深度剖析,深度理解js对象) 这算是酝酿很久的一篇文章了. JavaScript作为一个基于对象(没有类的概念)的语言,从入门到精通到放弃一直会被对象这个问题围绕 ...
- 原型模式和基于原型继承的js对象系统
像同样基于原型编程的Io语言一样,javascript在原型继承方面,实现原理和Io非常类似,javascript也遵守这些原则 所有数据都是对象 要得到一个对象,不是通过实例化类,而是找到一个对象作 ...
随机推荐
- Weblogic 9.2和10.3 改密码 一站完成
Weblogic 9.2和10.3可通用,只需修改参照如下配置即可: SET BEA_HOME=F:\beaSET JRE_HOME=%BEA_HOME%\jdk150_04\binSET LIB_H ...
- [转帖新闻]Windows 7时代即将终结:曾有多辉煌 如今就有多凄凉
Windows 7时代即将终结:曾有多辉煌 如今就有多凄凉 投递人 itwriter 发布于 2019-01-18 10:47 评论(4) 有834人阅读 [收藏] « » 文/屠敏 来源:CSDN( ...
- 使用Hexo搭建Github静态博客
1. 环境环境 1.1 安装Git 默认配置就好 1.2 安装node.js 下载:http://nodejs.org/download/ 安装时直接保持默认配置即可. 2. 配置Github 1.1 ...
- 3.27PSP及体会
首先,我还是第一次了解老师这种先喂鸡汤,再打鸡血的行为,大老板的出现让我有些措手不及,我 的 天 啊! 话说这周alpha版本实现,真的是好费脑筋,因为预定是4~6周的项目一周弄完,而且还是在拥有几个 ...
- delphi获取一个窗口的所有子窗口(包括嵌套)
unit Unit1; interface usesWindows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, ...
- 同步工具:CountDownLatch、CyclicBarrier和Semaphore
1. CountDownLatch 1.1 功能及使用场景 一个同步工具,使得一个或多个线程等待一组线程执行完成后再执行. 使用场景:等待一些前置任务执行完成后,再执行特定的功能.比如,系统启动时,各 ...
- Codeforces Round #276 (Div. 1) E. Sign on Fence (二分答案 主席树 区间合并)
链接:http://codeforces.com/contest/484/problem/E 题意: 给你n个数的,每个数代表高度: 再给出m个询问,每次询问[l,r]区间内连续w个数的最大的最小值: ...
- Web项目替换jar包中的文件的方法
经常遇到这样的问题,需要修改jar包中的方法.应该如何做? 1.有些很人性化的框架jar包,比如SpringSecurity,可以修改配置文件指定一个新建的类,让类实现Jar包中的对应的接口就好了. ...
- Studio启动的时候报错 Could not install Gradle distribution from
安装了studio以后,直接点击[Start]报如下错,绝对不是网络的问题. Could not install Gradle distribution from 'https://services. ...
- Linux开机自动挂载存储的两种方式
登录服务器,给查看了下,发现确实是没有自动加载,df -h只能显示本地硬盘的分区,fdisk -l 还是能看到存储空间,这说明这个服务器连接存储是木有问题的. 输入history | grep mou ...